Commit f50fe13f authored by Ian Hickson's avatar Ian Hickson Committed by GitHub

Fix flip-flop behaviour with two competing taps (#10151)

parent ee1db835
...@@ -58,6 +58,8 @@ abstract class GestureRecognizer extends GestureArenaMember { ...@@ -58,6 +58,8 @@ abstract class GestureRecognizer extends GestureArenaMember {
/// Invoke a callback provided by the application, catching and logging any /// Invoke a callback provided by the application, catching and logging any
/// exceptions. /// exceptions.
///
/// The `name` argument is ignored except when reporting exceptions.
@protected @protected
T invokeCallback<T>(String name, RecognizerCallback<T> callback) { T invokeCallback<T>(String name, RecognizerCallback<T> callback) {
T result; T result;
...@@ -78,6 +80,9 @@ abstract class GestureRecognizer extends GestureArenaMember { ...@@ -78,6 +80,9 @@ abstract class GestureRecognizer extends GestureArenaMember {
} }
return result; return result;
} }
@override
String toString() => '$runtimeType#$hashCode';
} }
/// Base class for gesture recognizers that can only recognize one /// Base class for gesture recognizers that can only recognize one
...@@ -109,7 +114,8 @@ abstract class OneSequenceGestureRecognizer extends GestureRecognizer { ...@@ -109,7 +114,8 @@ abstract class OneSequenceGestureRecognizer extends GestureRecognizer {
@protected @protected
void didStopTrackingLastPointer(int pointer); void didStopTrackingLastPointer(int pointer);
/// Resolves this recognizer's participation in each gesture arena with the given disposition. /// Resolves this recognizer's participation in each gesture arena with the
/// given disposition.
@protected @protected
@mustCallSuper @mustCallSuper
void resolve(GestureDisposition disposition) { void resolve(GestureDisposition disposition) {
...@@ -277,7 +283,7 @@ abstract class PrimaryPointerGestureRecognizer extends OneSequenceGestureRecogni ...@@ -277,7 +283,7 @@ abstract class PrimaryPointerGestureRecognizer extends OneSequenceGestureRecogni
@override @override
void rejectGesture(int pointer) { void rejectGesture(int pointer) {
if (pointer == primaryPointer) { if (pointer == primaryPointer && state == GestureRecognizerState.possible) {
_stopTimer(); _stopTimer();
state = GestureRecognizerState.defunct; state = GestureRecognizerState.defunct;
} }
...@@ -285,6 +291,7 @@ abstract class PrimaryPointerGestureRecognizer extends OneSequenceGestureRecogni ...@@ -285,6 +291,7 @@ abstract class PrimaryPointerGestureRecognizer extends OneSequenceGestureRecogni
@override @override
void didStopTrackingLastPointer(int pointer) { void didStopTrackingLastPointer(int pointer) {
assert(state != GestureRecognizerState.ready);
_stopTimer(); _stopTimer();
state = GestureRecognizerState.ready; state = GestureRecognizerState.ready;
} }
...@@ -307,4 +314,6 @@ abstract class PrimaryPointerGestureRecognizer extends OneSequenceGestureRecogni ...@@ -307,4 +314,6 @@ abstract class PrimaryPointerGestureRecognizer extends OneSequenceGestureRecogni
return offset.distance; return offset.distance;
} }
@override
String toString() => '$runtimeType#$hashCode($state)';
} }
...@@ -98,8 +98,10 @@ class TapGestureRecognizer extends PrimaryPointerGestureRecognizer { ...@@ -98,8 +98,10 @@ class TapGestureRecognizer extends PrimaryPointerGestureRecognizer {
@override @override
void resolve(GestureDisposition disposition) { void resolve(GestureDisposition disposition) {
if (_wonArenaForPrimaryPointer && disposition == GestureDisposition.rejected) { if (_wonArenaForPrimaryPointer && disposition == GestureDisposition.rejected) {
// This can happen if the superclass decides the primary pointer
// exceeded the touch slop, or if the recognizer is disposed.
if (onTapCancel != null) if (onTapCancel != null)
invokeCallback<Null>('onTapCancel', onTapCancel); // ignore: STRONG_MODE_INVALID_CAST_FUNCTION_EXPR, https://github.com/dart-lang/sdk/issues/27504 invokeCallback<Null>('spontaneous onTapCancel', onTapCancel);
_reset(); _reset();
} }
super.resolve(disposition); super.resolve(disposition);
...@@ -124,9 +126,10 @@ class TapGestureRecognizer extends PrimaryPointerGestureRecognizer { ...@@ -124,9 +126,10 @@ class TapGestureRecognizer extends PrimaryPointerGestureRecognizer {
void rejectGesture(int pointer) { void rejectGesture(int pointer) {
super.rejectGesture(pointer); super.rejectGesture(pointer);
if (pointer == primaryPointer) { if (pointer == primaryPointer) {
assert(state == GestureRecognizerState.defunct); // Another gesture won the arena.
assert(state != GestureRecognizerState.possible);
if (onTapCancel != null) if (onTapCancel != null)
invokeCallback<Null>('onTapCancel', onTapCancel); // ignore: STRONG_MODE_INVALID_CAST_FUNCTION_EXPR, https://github.com/dart-lang/sdk/issues/27504 invokeCallback<Null>('forced onTapCancel', onTapCancel);
_reset(); _reset();
} }
} }
...@@ -134,7 +137,7 @@ class TapGestureRecognizer extends PrimaryPointerGestureRecognizer { ...@@ -134,7 +137,7 @@ class TapGestureRecognizer extends PrimaryPointerGestureRecognizer {
void _checkDown() { void _checkDown() {
if (!_sentTapDown) { if (!_sentTapDown) {
if (onTapDown != null) if (onTapDown != null)
invokeCallback<Null>('onTapDown', () => onTapDown(new TapDownDetails(globalPosition: initialPosition))); // ignore: STRONG_MODE_INVALID_CAST_FUNCTION_EXPR, https://github.com/dart-lang/sdk/issues/27504 invokeCallback<Null>('onTapDown', () { onTapDown(new TapDownDetails(globalPosition: initialPosition)); });
_sentTapDown = true; _sentTapDown = true;
} }
} }
...@@ -143,9 +146,9 @@ class TapGestureRecognizer extends PrimaryPointerGestureRecognizer { ...@@ -143,9 +146,9 @@ class TapGestureRecognizer extends PrimaryPointerGestureRecognizer {
if (_wonArenaForPrimaryPointer && _finalPosition != null) { if (_wonArenaForPrimaryPointer && _finalPosition != null) {
resolve(GestureDisposition.accepted); resolve(GestureDisposition.accepted);
if (onTapUp != null) if (onTapUp != null)
invokeCallback<Null>('onTapUp', () => onTapUp(new TapUpDetails(globalPosition: _finalPosition))); // ignore: STRONG_MODE_INVALID_CAST_FUNCTION_EXPR, https://github.com/dart-lang/sdk/issues/27504 invokeCallback<Null>('onTapUp', () { onTapUp(new TapUpDetails(globalPosition: _finalPosition)); });
if (onTap != null) if (onTap != null)
invokeCallback<Null>('onTap', onTap); // ignore: STRONG_MODE_INVALID_CAST_FUNCTION_EXPR, https://github.com/dart-lang/sdk/issues/27504 invokeCallback<Null>('onTap', onTap);
_reset(); _reset();
} }
} }
......
...@@ -284,4 +284,75 @@ void main() { ...@@ -284,4 +284,75 @@ void main() {
FlutterError.onError = previousErrorHandler; FlutterError.onError = previousErrorHandler;
tap.dispose(); tap.dispose();
}); });
testGesture('No duplicate tap events', (GestureTester tester) {
final TapGestureRecognizer tapA = new TapGestureRecognizer();
final TapGestureRecognizer tapB = new TapGestureRecognizer();
final List<String> log = <String>[];
tapA.onTapDown = (TapDownDetails details) { log.add('tapA onTapDown'); };
tapA.onTapUp = (TapUpDetails details) { log.add('tapA onTapUp'); };
tapA.onTap = () { log.add('tapA onTap'); };
tapA.onTapCancel = () { log.add('tapA onTapCancel'); };
tapB.onTapDown = (TapDownDetails details) { log.add('tapB onTapDown'); };
tapB.onTapUp = (TapUpDetails details) { log.add('tapB onTapUp'); };
tapB.onTap = () { log.add('tapB onTap'); };
tapB.onTapCancel = () { log.add('tapB onTapCancel'); };
log.add('start');
tapA.addPointer(down1);
log.add('added 1 to A');
tapB.addPointer(down1);
log.add('added 1 to B');
tester.closeArena(1);
log.add('closed 1');
tester.route(down1);
log.add('routed 1 down');
tester.route(up1);
log.add('routed 1 up');
GestureBinding.instance.gestureArena.sweep(1);
log.add('swept 1');
tapA.addPointer(down2);
log.add('down 2 to A');
tapB.addPointer(down2);
log.add('down 2 to B');
tester.closeArena(2);
log.add('closed 2');
tester.route(down2);
log.add('routed 2 down');
tester.route(up2);
log.add('routed 2 up');
GestureBinding.instance.gestureArena.sweep(2);
log.add('swept 2');
tapA.dispose();
log.add('disposed A');
tapB.dispose();
log.add('disposed B');
expect(log, <String>[
'start',
'added 1 to A',
'added 1 to B',
'closed 1',
'routed 1 down',
'routed 1 up',
'tapA onTapDown',
'tapA onTapUp',
'tapA onTap',
'tapB onTapCancel',
'swept 1',
'down 2 to A',
'down 2 to B',
'closed 2',
'routed 2 down',
'routed 2 up',
'tapA onTapDown',
'tapA onTapUp',
'tapA onTap',
'tapB onTapCancel',
'swept 2',
'disposed A',
'disposed B',
]);
});
} }
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