Unverified Commit 750ad328 authored by Bruno Leroux's avatar Bruno Leroux Committed by GitHub

Fix DraggableScrollableSheet leaks Ticker (#102916)

parent 077e7e18
...@@ -795,8 +795,7 @@ class _DraggableScrollableSheetScrollController extends ScrollController { ...@@ -795,8 +795,7 @@ class _DraggableScrollableSheetScrollController extends ScrollController {
/// See also: /// See also:
/// ///
/// * [_DraggableScrollableSheetScrollController], which uses this as its [ScrollPosition]. /// * [_DraggableScrollableSheetScrollController], which uses this as its [ScrollPosition].
class _DraggableScrollableSheetScrollPosition class _DraggableScrollableSheetScrollPosition extends ScrollPositionWithSingleContext {
extends ScrollPositionWithSingleContext {
_DraggableScrollableSheetScrollPosition({ _DraggableScrollableSheetScrollPosition({
required super.physics, required super.physics,
required super.context, required super.context,
...@@ -805,16 +804,18 @@ class _DraggableScrollableSheetScrollPosition ...@@ -805,16 +804,18 @@ class _DraggableScrollableSheetScrollPosition
}); });
VoidCallback? _dragCancelCallback; VoidCallback? _dragCancelCallback;
VoidCallback? _ballisticCancelCallback;
final _DraggableSheetExtent Function() getExtent; final _DraggableSheetExtent Function() getExtent;
final Set<AnimationController> _ballisticControllers = <AnimationController>{};
bool get listShouldScroll => pixels > 0.0; bool get listShouldScroll => pixels > 0.0;
_DraggableSheetExtent get extent => getExtent(); _DraggableSheetExtent get extent => getExtent();
@override @override
void beginActivity(ScrollActivity? newActivity) { void beginActivity(ScrollActivity? newActivity) {
// Cancel the running ballistic simulation, if there is one. // Cancel the running ballistic simulations
_ballisticCancelCallback?.call(); for (final AnimationController ballisticController in _ballisticControllers) {
ballisticController.stop();
}
super.beginActivity(newActivity); super.beginActivity(newActivity);
} }
...@@ -852,8 +853,10 @@ class _DraggableScrollableSheetScrollPosition ...@@ -852,8 +853,10 @@ class _DraggableScrollableSheetScrollPosition
@override @override
void dispose() { void dispose() {
// Stop the animation before dispose. for (final AnimationController ballisticController in _ballisticControllers) {
_ballisticCancelCallback?.call(); ballisticController.dispose();
}
_ballisticControllers.clear();
super.dispose(); super.dispose();
} }
...@@ -873,10 +876,11 @@ class _DraggableScrollableSheetScrollPosition ...@@ -873,10 +876,11 @@ class _DraggableScrollableSheetScrollPosition
if (extent.snap) { if (extent.snap) {
// Snap is enabled, simulate snapping instead of clamping scroll. // Snap is enabled, simulate snapping instead of clamping scroll.
simulation = _SnappingSimulation( simulation = _SnappingSimulation(
position: extent.currentPixels, position: extent.currentPixels,
initialVelocity: velocity, initialVelocity: velocity,
pixelSnapSize: extent.pixelSnapSizes, pixelSnapSize: extent.pixelSnapSizes,
tolerance: physics.tolerance); tolerance: physics.tolerance,
);
} else { } else {
// The iOS bouncing simulation just isn't right here - once we delegate // The iOS bouncing simulation just isn't right here - once we delegate
// the ballistic back to the ScrollView, it will use the right simulation. // the ballistic back to the ScrollView, it will use the right simulation.
...@@ -892,9 +896,8 @@ class _DraggableScrollableSheetScrollPosition ...@@ -892,9 +896,8 @@ class _DraggableScrollableSheetScrollPosition
debugLabel: objectRuntimeType(this, '_DraggableScrollableSheetPosition'), debugLabel: objectRuntimeType(this, '_DraggableScrollableSheetPosition'),
vsync: context.vsync, vsync: context.vsync,
); );
// Stop the ballistic animation if a new activity starts. _ballisticControllers.add(ballisticController);
// See: [beginActivity].
_ballisticCancelCallback = ballisticController.stop;
double lastPosition = extent.currentPixels; double lastPosition = extent.currentPixels;
void tick() { void tick() {
final double delta = ballisticController.value - lastPosition; final double delta = ballisticController.value - lastPosition;
...@@ -916,8 +919,10 @@ class _DraggableScrollableSheetScrollPosition ...@@ -916,8 +919,10 @@ class _DraggableScrollableSheetScrollPosition
..addListener(tick) ..addListener(tick)
..animateWith(simulation).whenCompleteOrCancel( ..animateWith(simulation).whenCompleteOrCancel(
() { () {
_ballisticCancelCallback = null; if (_ballisticControllers.contains(ballisticController)) {
ballisticController.dispose(); _ballisticControllers.remove(ballisticController);
ballisticController.dispose();
}
}, },
); );
} }
......
...@@ -326,7 +326,59 @@ void main() { ...@@ -326,7 +326,59 @@ void main() {
expect(find.text('Item 70'), findsNothing); expect(find.text('Item 70'), findsNothing);
}, variant: TargetPlatformVariant.all()); }, variant: TargetPlatformVariant.all());
debugDefaultTargetPlatformOverride = null; testWidgets('Ballistic animation on fling should not leak Ticker', (WidgetTester tester) async {
// Regression test for https://github.com/flutter/flutter/issues/101061
await tester.pumpWidget(
Directionality(
textDirection: TextDirection.ltr,
child: MediaQuery(
data: const MediaQueryData(),
child: Align(
alignment: Alignment.bottomCenter,
child: DraggableScrollableSheet(
initialChildSize: 0.8,
minChildSize: 0.2,
maxChildSize: 0.9,
expand: false,
builder: (_, ScrollController scrollController) {
return ListView.separated(
physics: const BouncingScrollPhysics(),
controller: scrollController,
separatorBuilder: (_, __) => const Divider(),
itemCount: 100,
itemBuilder: (_, int index) => SizedBox(
height: 100,
child: ColoredBox(
color: Colors.primaries[index % Colors.primaries.length],
child: Text('Item $index'),
),
),
);
},
),
),
),
),
);
await tester.flingFrom(
tester.getCenter(find.text('Item 1')),
const Offset(0, 50),
10000,
);
// Pumps several times to let the DraggableScrollableSheet react to scroll position changes.
const int numberOfPumpsBeforeError = 22;
for (int i = 0; i < numberOfPumpsBeforeError; i++) {
await tester.pump(const Duration(milliseconds: 10));
}
// Dispose the DraggableScrollableSheet
await tester.pumpWidget(const SizedBox.shrink());
// When a Ticker leaks an exception is thrown
expect(tester.takeException(), isNull);
});
}); });
testWidgets('Does not snap away from initial child on build', (WidgetTester tester) async { testWidgets('Does not snap away from initial child on build', (WidgetTester tester) async {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment