mirror of
https://github.com/flutter/flutter.git
synced 2026-02-20 02:29:02 +08:00
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
184 lines
7.3 KiB
Dart
184 lines
7.3 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_test/flutter_test.dart';
|
|
|
|
import 'rendering_tester.dart';
|
|
|
|
class TestTree {
|
|
TestTree() {
|
|
// incoming constraints are tight 800x600
|
|
root = RenderPositionedBox(
|
|
child: RenderConstrainedBox(
|
|
additionalConstraints: const BoxConstraints.tightFor(width: 800.0),
|
|
// Place the child to be evaluated within both a repaint boundary and a
|
|
// layout-root element (in this case a tightly constrained box). Otherwise
|
|
// the act of transplanting the root into a new container will cause the
|
|
// relayout/repaint of the new parent node to satisfy the test.
|
|
child: RenderRepaintBoundary(
|
|
child: RenderConstrainedBox(
|
|
additionalConstraints: const BoxConstraints.tightFor(height: 20.0, width: 20.0),
|
|
child: RenderRepaintBoundary(
|
|
child: RenderCustomPaint(
|
|
painter: TestCallbackPainter(
|
|
onPaint: () {
|
|
painted = true;
|
|
},
|
|
),
|
|
child: RenderPositionedBox(
|
|
child: child = RenderConstrainedBox(
|
|
additionalConstraints: const BoxConstraints.tightFor(height: 20.0, width: 20.0),
|
|
child: RenderSemanticsAnnotations(
|
|
textDirection: TextDirection.ltr,
|
|
properties: const SemanticsProperties(label: 'Hello there foo'),
|
|
),
|
|
),
|
|
),
|
|
),
|
|
),
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
late RenderBox root;
|
|
late RenderConstrainedBox child;
|
|
bool painted = false;
|
|
}
|
|
|
|
class MutableCompositor extends RenderProxyBox {
|
|
MutableCompositor({required RenderBox child}) : super(child);
|
|
bool _alwaysComposite = false;
|
|
@override
|
|
bool get alwaysNeedsCompositing => _alwaysComposite;
|
|
}
|
|
|
|
class TestCompositingBitsTree {
|
|
TestCompositingBitsTree() {
|
|
// incoming constraints are tight 800x600
|
|
root = RenderPositionedBox(
|
|
child: RenderConstrainedBox(
|
|
additionalConstraints: const BoxConstraints.tightFor(width: 800.0),
|
|
// Place the child to be evaluated within a repaint boundary. Otherwise
|
|
// the act of transplanting the root into a new container will cause the
|
|
// repaint of the new parent node to satisfy the test.
|
|
child: RenderRepaintBoundary(
|
|
child: compositor = MutableCompositor(
|
|
child: RenderCustomPaint(
|
|
painter: TestCallbackPainter(
|
|
onPaint: () {
|
|
painted = true;
|
|
},
|
|
),
|
|
child: child = RenderConstrainedBox(
|
|
additionalConstraints: const BoxConstraints.tightFor(height: 20.0, width: 20.0),
|
|
),
|
|
),
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
late RenderBox root;
|
|
late MutableCompositor compositor;
|
|
late RenderConstrainedBox child;
|
|
bool painted = false;
|
|
}
|
|
|
|
void main() {
|
|
TestRenderingFlutterBinding.ensureInitialized();
|
|
|
|
test('objects can be detached and re-attached: layout', () {
|
|
final TestTree testTree = TestTree();
|
|
// Lay out
|
|
layout(testTree.root);
|
|
expect(testTree.child.size, equals(const Size(20.0, 20.0)));
|
|
// Remove testTree from the custom render view
|
|
TestRenderingFlutterBinding.instance.renderView.child = null;
|
|
expect(testTree.child.owner, isNull);
|
|
// Dirty one of the elements
|
|
testTree.child.additionalConstraints = const BoxConstraints.tightFor(height: 5.0, width: 5.0);
|
|
// Lay out again
|
|
layout(testTree.root);
|
|
expect(testTree.child.size, equals(const Size(5.0, 5.0)));
|
|
});
|
|
test('objects can be detached and re-attached: compositingBits', () {
|
|
final TestCompositingBitsTree testTree = TestCompositingBitsTree();
|
|
// Lay out, composite, and paint
|
|
layout(testTree.root, phase: EnginePhase.paint);
|
|
expect(testTree.painted, isTrue);
|
|
// Remove testTree from the custom render view
|
|
TestRenderingFlutterBinding.instance.renderView.child = null;
|
|
expect(testTree.child.owner, isNull);
|
|
// Dirty one of the elements
|
|
testTree.compositor._alwaysComposite = true;
|
|
testTree.child.markNeedsCompositingBitsUpdate();
|
|
testTree.painted = false;
|
|
// Lay out, composite, and paint again
|
|
layout(testTree.root, phase: EnginePhase.paint);
|
|
expect(testTree.painted, isTrue);
|
|
});
|
|
test('objects can be detached and re-attached: paint', () {
|
|
final TestTree testTree = TestTree();
|
|
// Lay out, composite, and paint
|
|
layout(testTree.root, phase: EnginePhase.paint);
|
|
expect(testTree.painted, isTrue);
|
|
// Remove testTree from the custom render view
|
|
TestRenderingFlutterBinding.instance.renderView.child = null;
|
|
expect(testTree.child.owner, isNull);
|
|
// Dirty one of the elements
|
|
testTree.child.markNeedsPaint();
|
|
testTree.painted = false;
|
|
// Lay out, composite, and paint again
|
|
layout(testTree.root, phase: EnginePhase.paint);
|
|
expect(testTree.painted, isTrue);
|
|
});
|
|
test('objects can be detached and re-attached: semantics (no change)', () {
|
|
final TestTree testTree = TestTree();
|
|
int semanticsUpdateCount = 0;
|
|
final SemanticsHandle semanticsHandle = TestRenderingFlutterBinding.instance.ensureSemantics();
|
|
TestRenderingFlutterBinding.instance.pipelineOwner.semanticsOwner!.addListener(() {
|
|
++semanticsUpdateCount;
|
|
});
|
|
// Lay out, composite, paint, and update semantics
|
|
layout(testTree.root, phase: EnginePhase.flushSemantics);
|
|
expect(semanticsUpdateCount, 1);
|
|
// Remove testTree from the custom render view
|
|
TestRenderingFlutterBinding.instance.renderView.child = null;
|
|
expect(testTree.child.owner, isNull);
|
|
// Dirty one of the elements
|
|
semanticsUpdateCount = 0;
|
|
testTree.child.markNeedsSemanticsUpdate();
|
|
expect(semanticsUpdateCount, 0);
|
|
// Lay out, composite, paint, and update semantics again
|
|
layout(testTree.root, phase: EnginePhase.flushSemantics);
|
|
expect(semanticsUpdateCount, 0); // no semantics have changed.
|
|
semanticsHandle.dispose();
|
|
});
|
|
test('objects can be detached and re-attached: semantics (with change)', () {
|
|
final TestTree testTree = TestTree();
|
|
int semanticsUpdateCount = 0;
|
|
final SemanticsHandle semanticsHandle = TestRenderingFlutterBinding.instance.ensureSemantics();
|
|
TestRenderingFlutterBinding.instance.pipelineOwner.semanticsOwner!.addListener(() {
|
|
++semanticsUpdateCount;
|
|
});
|
|
// Lay out, composite, paint, and update semantics
|
|
layout(testTree.root, phase: EnginePhase.flushSemantics);
|
|
expect(semanticsUpdateCount, 1);
|
|
// Remove testTree from the custom render view
|
|
TestRenderingFlutterBinding.instance.renderView.child = null;
|
|
expect(testTree.child.owner, isNull);
|
|
// Dirty one of the elements
|
|
semanticsUpdateCount = 0;
|
|
testTree.child.additionalConstraints = const BoxConstraints.tightFor(height: 20.0, width: 30.0);
|
|
testTree.child.markNeedsSemanticsUpdate();
|
|
expect(semanticsUpdateCount, 0);
|
|
// Lay out, composite, paint, and update semantics again
|
|
layout(testTree.root, phase: EnginePhase.flushSemantics);
|
|
expect(semanticsUpdateCount, 1); // semantics have changed.
|
|
semanticsHandle.dispose();
|
|
});
|
|
}
|