Skip to content

Commit 8203b6e

Browse files
angelosilvestreTytaniumDev
authored andcommitted
Make DragGestureRecognizer abstract methods public (flutter#151627)
Resolves flutter#151446 `DragGestureRecognizer` defines several private abstract methods that are implemented by its subclasses. In the **super_editor** package, we'd like to extend `PanGestureRecognizer` to make it more aggressive, so it can win the gesture arena when placed inside a `CustomScrollview`. However, since we can't override private methods, tweaking this single function would involve copying the entire `DragGestureRecognizer` interface and its `PanGestureRecognizer` implementation. <br> Methods that were updated in this PR: | Method | Rationale | |---|---| | `_hasSufficientGlobalDistanceToAccept` | This is the most important method for us. Overriding this method allows tweaking the PanGestureRecognizer to be more aggressive. | | `_considerFling` | In **super_editor** we use the PanGestureRecognizer, but we want the fling gesture to behave as if it was a VerticalDragRecognizer. We'll use the fling gesture just to scroll vertically. | | `_finalPosition` | I added a getter to be able to access it inside `_considerFling`. | | `_globalDistanceMoved` | I added a getter to be able to access it inside `_hasSufficientGlobalDistanceToAccept`. |
1 parent 81dabea commit 8203b6e

File tree

2 files changed

+144
-25
lines changed

2 files changed

+144
-25
lines changed

packages/flutter/lib/src/gestures/monodrag.dart

+53-25
Original file line numberDiff line numberDiff line change
@@ -75,7 +75,7 @@ typedef GestureVelocityTrackerBuilder = VelocityTracker Function(PointerEvent ev
7575
/// * [HorizontalDragGestureRecognizer], for left and right drags.
7676
/// * [VerticalDragGestureRecognizer], for up and down drags.
7777
/// * [PanGestureRecognizer], for drags that are not locked to a single axis.
78-
abstract class DragGestureRecognizer extends OneSequenceGestureRecognizer {
78+
sealed class DragGestureRecognizer extends OneSequenceGestureRecognizer {
7979
/// Initialize the object.
8080
///
8181
/// {@macro flutter.gestures.GestureRecognizer.supportedDevices}
@@ -287,7 +287,14 @@ abstract class DragGestureRecognizer extends OneSequenceGestureRecognizer {
287287
_DragState _state = _DragState.ready;
288288
late OffsetPair _initialPosition;
289289
late OffsetPair _pendingDragOffset;
290-
late OffsetPair _finalPosition;
290+
291+
/// The local and global offsets of the last pointer event received.
292+
///
293+
/// It is used to create the [DragEndDetails], which provides information about
294+
/// the end of a drag gesture.
295+
OffsetPair get lastPosition => _lastPosition;
296+
late OffsetPair _lastPosition;
297+
291298
Duration? _lastPendingEventTimestamp;
292299

293300
/// When asserts are enabled, returns the last tracked pending event timestamp
@@ -316,6 +323,7 @@ abstract class DragGestureRecognizer extends OneSequenceGestureRecognizer {
316323
///
317324
/// If drag is only allowed along a defined axis, this value may be negative to
318325
/// differentiate the direction of the drag.
326+
double get globalDistanceMoved => _globalDistanceMoved;
319327
late double _globalDistanceMoved;
320328

321329
/// Determines if a gesture is a fling or not based on velocity.
@@ -330,16 +338,36 @@ abstract class DragGestureRecognizer extends OneSequenceGestureRecognizer {
330338
/// A fling calls its gesture end callback with a velocity, allowing the
331339
/// provider of the callback to respond by carrying the gesture forward with
332340
/// inertia, for example.
333-
DragEndDetails? _considerFling(VelocityEstimate estimate, PointerDeviceKind kind);
341+
DragEndDetails? considerFling(VelocityEstimate estimate, PointerDeviceKind kind);
334342

343+
/// Returns the effective delta that should be considered for the incoming [delta].
344+
///
345+
/// The delta received by an event might contain both the x and y components
346+
/// greater than zero, and an one-axis drag recognizer only cares about one
347+
/// of them.
348+
///
349+
/// For example, a [VerticalDragGestureRecognizer], would return an [Offset]
350+
/// with the x component set to 0.0, because it only cares about the y component.
335351
Offset _getDeltaForDetails(Offset delta);
352+
353+
/// Returns the value for the primary axis from the given [value].
354+
///
355+
/// For example, a [VerticalDragGestureRecognizer] would return the y
356+
/// component, while a [HorizontalDragGestureRecognizer] would return
357+
/// the x component.
358+
///
359+
/// Returns `null` if the recognizer does not have a primary axis.
336360
double? _getPrimaryValueFromOffset(Offset value);
337361

338362
/// The axis (horizontal or vertical) corresponding to the primary drag direction.
339363
///
340364
/// The [PanGestureRecognizer] returns null.
341365
_DragDirection? _getPrimaryDragAxis() => null;
342-
bool _hasSufficientGlobalDistanceToAccept(PointerDeviceKind pointerDeviceKind, double? deviceTouchSlop);
366+
367+
/// Whether the [globalDistanceMoved] is big enough to accept the gesture.
368+
///
369+
/// If this method returns `true`, it means this recognizer should declare win in the gesture arena.
370+
bool hasSufficientGlobalDistanceToAccept(PointerDeviceKind pointerDeviceKind, double? deviceTouchSlop);
343371
bool _hasDragThresholdBeenMet = false;
344372

345373
final Map<int, VelocityTracker> _velocityTrackers = <int, VelocityTracker>{};
@@ -380,7 +408,7 @@ abstract class DragGestureRecognizer extends OneSequenceGestureRecognizer {
380408
case _DragState.ready:
381409
_state = _DragState.possible;
382410
_initialPosition = OffsetPair(global: event.position, local: event.localPosition);
383-
_finalPosition = _initialPosition;
411+
_lastPosition = _initialPosition;
384412
_pendingDragOffset = OffsetPair.zero;
385413
_globalDistanceMoved = 0.0;
386414
_lastPendingEventTimestamp = event.timeStamp;
@@ -629,7 +657,7 @@ abstract class DragGestureRecognizer extends OneSequenceGestureRecognizer {
629657
final Offset localDelta = (event is PointerMoveEvent) ? event.localDelta : (event as PointerPanZoomUpdateEvent).localPanDelta;
630658
final Offset position = (event is PointerMoveEvent) ? event.position : (event.position + (event as PointerPanZoomUpdateEvent).pan);
631659
final Offset localPosition = (event is PointerMoveEvent) ? event.localPosition : (event.localPosition + (event as PointerPanZoomUpdateEvent).localPan);
632-
_finalPosition = OffsetPair(local: localPosition, global: position);
660+
_lastPosition = OffsetPair(local: localPosition, global: position);
633661
final Offset resolvedDelta = _resolveLocalDeltaForMultitouch(event.pointer, localDelta);
634662
switch (_state) {
635663
case _DragState.ready || _DragState.possible:
@@ -643,7 +671,7 @@ abstract class DragGestureRecognizer extends OneSequenceGestureRecognizer {
643671
untransformedDelta: movedLocally,
644672
untransformedEndPosition: localPosition
645673
).distance * (_getPrimaryValueFromOffset(movedLocally) ?? 1).sign;
646-
if (_hasSufficientGlobalDistanceToAccept(event.kind, gestureSettings?.touchSlop)) {
674+
if (hasSufficientGlobalDistanceToAccept(event.kind, gestureSettings?.touchSlop)) {
647675
_hasDragThresholdBeenMet = true;
648676
if (_acceptedActivePointers.contains(event.pointer)) {
649677
_checkDrag(event.pointer);
@@ -823,15 +851,15 @@ abstract class DragGestureRecognizer extends OneSequenceGestureRecognizer {
823851
if (estimate == null) {
824852
debugReport = () => 'Could not estimate velocity.';
825853
} else {
826-
details = _considerFling(estimate, tracker.kind);
854+
details = considerFling(estimate, tracker.kind);
827855
debugReport = (details != null)
828856
? () => '$estimate; fling at ${details!.velocity}.'
829857
: () => '$estimate; judged to not be a fling.';
830858
}
831859
details ??= DragEndDetails(
832860
primaryVelocity: 0.0,
833-
globalPosition: _finalPosition.global,
834-
localPosition: _finalPosition.local,
861+
globalPosition: _lastPosition.global,
862+
localPosition: _lastPosition.local,
835863
);
836864

837865
invokeCallback<void>('onEnd', () => onEnd!(details!), debugReport: debugReport);
@@ -883,7 +911,7 @@ class VerticalDragGestureRecognizer extends DragGestureRecognizer {
883911
}
884912

885913
@override
886-
DragEndDetails? _considerFling(VelocityEstimate estimate, PointerDeviceKind kind) {
914+
DragEndDetails? considerFling(VelocityEstimate estimate, PointerDeviceKind kind) {
887915
if (!isFlingGesture(estimate, kind)) {
888916
return null;
889917
}
@@ -892,14 +920,14 @@ class VerticalDragGestureRecognizer extends DragGestureRecognizer {
892920
return DragEndDetails(
893921
velocity: Velocity(pixelsPerSecond: Offset(0, dy)),
894922
primaryVelocity: dy,
895-
globalPosition: _finalPosition.global,
896-
localPosition: _finalPosition.local,
923+
globalPosition: lastPosition.global,
924+
localPosition: lastPosition.local,
897925
);
898926
}
899927

900928
@override
901-
bool _hasSufficientGlobalDistanceToAccept(PointerDeviceKind pointerDeviceKind, double? deviceTouchSlop) {
902-
return _globalDistanceMoved.abs() > computeHitSlop(pointerDeviceKind, gestureSettings);
929+
bool hasSufficientGlobalDistanceToAccept(PointerDeviceKind pointerDeviceKind, double? deviceTouchSlop) {
930+
return globalDistanceMoved.abs() > computeHitSlop(pointerDeviceKind, gestureSettings);
903931
}
904932

905933
@override
@@ -943,7 +971,7 @@ class HorizontalDragGestureRecognizer extends DragGestureRecognizer {
943971
}
944972

945973
@override
946-
DragEndDetails? _considerFling(VelocityEstimate estimate, PointerDeviceKind kind) {
974+
DragEndDetails? considerFling(VelocityEstimate estimate, PointerDeviceKind kind) {
947975
if (!isFlingGesture(estimate, kind)) {
948976
return null;
949977
}
@@ -952,14 +980,14 @@ class HorizontalDragGestureRecognizer extends DragGestureRecognizer {
952980
return DragEndDetails(
953981
velocity: Velocity(pixelsPerSecond: Offset(dx, 0)),
954982
primaryVelocity: dx,
955-
globalPosition: _finalPosition.global,
956-
localPosition: _finalPosition.local,
983+
globalPosition: _lastPosition.global,
984+
localPosition: _lastPosition.local,
957985
);
958986
}
959987

960988
@override
961-
bool _hasSufficientGlobalDistanceToAccept(PointerDeviceKind pointerDeviceKind, double? deviceTouchSlop) {
962-
return _globalDistanceMoved.abs() > computeHitSlop(pointerDeviceKind, gestureSettings);
989+
bool hasSufficientGlobalDistanceToAccept(PointerDeviceKind pointerDeviceKind, double? deviceTouchSlop) {
990+
return globalDistanceMoved.abs() > computeHitSlop(pointerDeviceKind, gestureSettings);
963991
}
964992

965993
@override
@@ -1001,22 +1029,22 @@ class PanGestureRecognizer extends DragGestureRecognizer {
10011029
}
10021030

10031031
@override
1004-
DragEndDetails? _considerFling(VelocityEstimate estimate, PointerDeviceKind kind) {
1032+
DragEndDetails? considerFling(VelocityEstimate estimate, PointerDeviceKind kind) {
10051033
if (!isFlingGesture(estimate, kind)) {
10061034
return null;
10071035
}
10081036
final Velocity velocity = Velocity(pixelsPerSecond: estimate.pixelsPerSecond)
10091037
.clampMagnitude(minFlingVelocity ?? kMinFlingVelocity, maxFlingVelocity ?? kMaxFlingVelocity);
10101038
return DragEndDetails(
10111039
velocity: velocity,
1012-
globalPosition: _finalPosition.global,
1013-
localPosition: _finalPosition.local,
1040+
globalPosition: lastPosition.global,
1041+
localPosition: lastPosition.local,
10141042
);
10151043
}
10161044

10171045
@override
1018-
bool _hasSufficientGlobalDistanceToAccept(PointerDeviceKind pointerDeviceKind, double? deviceTouchSlop) {
1019-
return _globalDistanceMoved.abs() > computePanSlop(pointerDeviceKind, gestureSettings);
1046+
bool hasSufficientGlobalDistanceToAccept(PointerDeviceKind pointerDeviceKind, double? deviceTouchSlop) {
1047+
return globalDistanceMoved.abs() > computePanSlop(pointerDeviceKind, gestureSettings);
10201048
}
10211049

10221050
@override

packages/flutter/test/gestures/monodrag_test.dart

+91
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,10 @@
22
// Use of this source code is governed by a BSD-style license that can be
33
// found in the LICENSE file.
44

5+
import 'dart:ui';
6+
57
import 'package:flutter/gestures.dart';
8+
import 'package:flutter/material.dart';
69
import 'package:flutter_test/flutter_test.dart';
710

811
import 'gesture_tester.dart';
@@ -145,6 +148,58 @@ void main() {
145148
dragCallbacks.clear();
146149
});
147150

151+
testWidgets('DragGestureRecognizer can be subclassed to beat a CustomScrollView in the arena', (WidgetTester tester) async {
152+
final GlobalKey tapTargetKey = GlobalKey();
153+
bool wasPanStartCalled = false;
154+
155+
// Pump a tree with panable widget inside a CustomScrollView. The CustomScrollView
156+
// has a more aggresive drag recognizer that will typically beat other drag
157+
// recognizers in the arena. This pan recognizer uses a smaller threshold to
158+
// accept the gesture, that should make it win the arena.
159+
await tester.pumpWidget(
160+
MaterialApp(home:
161+
Scaffold(
162+
body: CustomScrollView(
163+
slivers: <Widget>[
164+
SliverToBoxAdapter(
165+
child: RawGestureDetector(
166+
behavior: HitTestBehavior.translucent,
167+
gestures: <Type, GestureRecognizerFactory>{
168+
_EagerPanGestureRecognizer: GestureRecognizerFactoryWithHandlers<_EagerPanGestureRecognizer>(
169+
() => _EagerPanGestureRecognizer(),
170+
(_EagerPanGestureRecognizer recognizer) {
171+
recognizer
172+
.onStart = (DragStartDetails details) => wasPanStartCalled = true;
173+
},
174+
),
175+
},
176+
child: SizedBox(
177+
key: tapTargetKey,
178+
width: 100,
179+
height: 100,
180+
),
181+
),
182+
),
183+
],
184+
),
185+
),
186+
),
187+
);
188+
189+
// Tap down on the tap target inside the gesture recognizer.
190+
final TestGesture gesture = await tester.startGesture(tester.getCenter(find.byKey(tapTargetKey)));
191+
await tester.pump();
192+
193+
// Move the pointer predominantly on the x-axis, with a y-axis movement that
194+
// is sufficient bigger so that both the CustomScrollScrollView and the
195+
// pan gesture recognizer want to accept the gesture.
196+
await gesture.moveBy(const Offset(30, kTouchSlop + 1));
197+
await tester.pump();
198+
199+
// Ensure our gesture recognizer won the arena.
200+
expect(wasPanStartCalled, isTrue);
201+
});
202+
148203
group('Recognizers on different button filters:', () {
149204
final List<String> recognized = <String>[];
150205
late HorizontalDragGestureRecognizer primaryRecognizer;
@@ -201,3 +256,39 @@ class MockHitTestTarget implements HitTestTarget {
201256
@override
202257
void handleEvent(PointerEvent event, HitTestEntry entry) { }
203258
}
259+
260+
/// A [PanGestureRecognizer] that tries to beat [VerticalDragGestureRecognizer] in the arena.
261+
///
262+
/// Typically, [VerticalDragGestureRecognizer] wins because it has a smaller threshold to
263+
/// accept the gesture. This recognizer uses the same threshold that [VerticalDragGestureRecognizer]
264+
/// uses.
265+
class _EagerPanGestureRecognizer extends PanGestureRecognizer {
266+
267+
@override
268+
bool hasSufficientGlobalDistanceToAccept(PointerDeviceKind pointerDeviceKind, double? deviceTouchSlop) {
269+
return globalDistanceMoved.abs() > computeHitSlop(pointerDeviceKind, gestureSettings);
270+
}
271+
272+
@override
273+
bool isFlingGesture(VelocityEstimate estimate, PointerDeviceKind kind) {
274+
final double minVelocity = minFlingVelocity ?? kMinFlingVelocity;
275+
final double minDistance = minFlingDistance ?? computeHitSlop(kind, gestureSettings);
276+
return estimate.pixelsPerSecond.distanceSquared > minVelocity * minVelocity &&
277+
estimate.offset.distanceSquared > minDistance * minDistance;
278+
}
279+
280+
@override
281+
DragEndDetails? considerFling(VelocityEstimate estimate, PointerDeviceKind kind) {
282+
if (!isFlingGesture(estimate, kind)) {
283+
return null;
284+
}
285+
final double maxVelocity = maxFlingVelocity ?? kMaxFlingVelocity;
286+
final double dy = clampDouble(estimate.pixelsPerSecond.dy, -maxVelocity, maxVelocity);
287+
return DragEndDetails(
288+
velocity: Velocity(pixelsPerSecond: Offset(0, dy)),
289+
primaryVelocity: dy,
290+
globalPosition: lastPosition.global,
291+
localPosition: lastPosition.local,
292+
);
293+
}
294+
}

0 commit comments

Comments
 (0)