Unverified Commit 7122a009 authored by Greg Spencer's avatar Greg Spencer Committed by GitHub

Fix some tap region bugs (#110398)

parent ee98003e
...@@ -406,7 +406,7 @@ class TapRegion extends SingleChildRenderObjectWidget { ...@@ -406,7 +406,7 @@ class TapRegion extends SingleChildRenderObjectWidget {
/// ///
/// * [TapRegion], a widget that inserts a [RenderTapRegion] into the render /// * [TapRegion], a widget that inserts a [RenderTapRegion] into the render
/// tree. /// tree.
class RenderTapRegion extends RenderProxyBox with Diagnosticable { class RenderTapRegion extends RenderProxyBox {
/// Creates a [RenderTapRegion]. /// Creates a [RenderTapRegion].
RenderTapRegion({ RenderTapRegion({
TapRegionRegistry? registry, TapRegionRegistry? registry,
...@@ -464,6 +464,12 @@ class RenderTapRegion extends RenderProxyBox with Diagnosticable { ...@@ -464,6 +464,12 @@ class RenderTapRegion extends RenderProxyBox with Diagnosticable {
Object? _groupId; Object? _groupId;
set groupId(Object? value) { set groupId(Object? value) {
if (_groupId != value) { if (_groupId != value) {
// If the group changes, we need to unregister and re-register under the
// new group. The re-registration happens automatically in layout().
if (_isRegistered) {
_registry!.unregisterTapRegion(this);
_isRegistered = false;
}
_groupId = value; _groupId = value;
markNeedsLayout(); markNeedsLayout();
} }
...@@ -515,7 +521,7 @@ class RenderTapRegion extends RenderProxyBox with Diagnosticable { ...@@ -515,7 +521,7 @@ class RenderTapRegion extends RenderProxyBox with Diagnosticable {
@override @override
void debugFillProperties(DiagnosticPropertiesBuilder properties) { void debugFillProperties(DiagnosticPropertiesBuilder properties) {
super.debugFillProperties(properties); super.debugFillProperties(properties);
properties.add(DiagnosticsProperty<Object?>('debugLabel', debugLabel, defaultValue: null)); properties.add(DiagnosticsProperty<String?>('debugLabel', debugLabel, defaultValue: null));
properties.add(DiagnosticsProperty<Object?>('groupId', groupId, defaultValue: null)); properties.add(DiagnosticsProperty<Object?>('groupId', groupId, defaultValue: null));
properties.add(FlagProperty('enabled', value: enabled, ifFalse: 'DISABLED', defaultValue: true)); properties.add(FlagProperty('enabled', value: enabled, ifFalse: 'DISABLED', defaultValue: true));
} }
......
...@@ -9,7 +9,7 @@ import 'package:flutter_test/flutter_test.dart'; ...@@ -9,7 +9,7 @@ import 'package:flutter_test/flutter_test.dart';
void main() { void main() {
testWidgets('TapRegionSurface detects outside taps', (WidgetTester tester) async { testWidgets('TapRegionSurface detects outside taps', (WidgetTester tester) async {
final Set<String> clickedOutside = <String>{}; final Set<String> tappedOutside = <String>{};
await tester.pumpWidget( await tester.pumpWidget(
Directionality( Directionality(
textDirection: TextDirection.ltr, textDirection: TextDirection.ltr,
...@@ -22,21 +22,21 @@ void main() { ...@@ -22,21 +22,21 @@ void main() {
const Text('Outside'), const Text('Outside'),
TapRegion( TapRegion(
onTapOutside: (PointerEvent event) { onTapOutside: (PointerEvent event) {
clickedOutside.add('No Group'); tappedOutside.add('No Group');
}, },
child: const Text('No Group'), child: const Text('No Group'),
), ),
TapRegion( TapRegion(
groupId: 1, groupId: 1,
onTapOutside: (PointerEvent event) { onTapOutside: (PointerEvent event) {
clickedOutside.add('Group 1 A'); tappedOutside.add('Group 1 A');
}, },
child: const Text('Group 1 A'), child: const Text('Group 1 A'),
), ),
TapRegion( TapRegion(
groupId: 1, groupId: 1,
onTapOutside: (PointerEvent event) { onTapOutside: (PointerEvent event) {
clickedOutside.add('Group 1 B'); tappedOutside.add('Group 1 B');
}, },
child: const Text('Group 1 B'), child: const Text('Group 1 B'),
), ),
...@@ -59,48 +59,48 @@ void main() { ...@@ -59,48 +59,48 @@ void main() {
await gesture.removePointer(); await gesture.removePointer();
} }
expect(clickedOutside, isEmpty); expect(tappedOutside, isEmpty);
await click(find.text('No Group')); await click(find.text('No Group'));
expect( expect(
clickedOutside, tappedOutside,
unorderedEquals(<String>{ unorderedEquals(<String>{
'Group 1 A', 'Group 1 A',
'Group 1 B', 'Group 1 B',
})); }));
clickedOutside.clear(); tappedOutside.clear();
await click(find.text('Group 1 A')); await click(find.text('Group 1 A'));
expect( expect(
clickedOutside, tappedOutside,
equals(<String>{ equals(<String>{
'No Group', 'No Group',
})); }));
clickedOutside.clear(); tappedOutside.clear();
await click(find.text('Group 1 B')); await click(find.text('Group 1 B'));
expect( expect(
clickedOutside, tappedOutside,
equals(<String>{ equals(<String>{
'No Group', 'No Group',
})); }));
clickedOutside.clear(); tappedOutside.clear();
await click(find.text('Outside')); await click(find.text('Outside'));
expect( expect(
clickedOutside, tappedOutside,
unorderedEquals(<String>{ unorderedEquals(<String>{
'No Group', 'No Group',
'Group 1 A', 'Group 1 A',
'Group 1 B', 'Group 1 B',
})); }));
clickedOutside.clear(); tappedOutside.clear();
await click(find.text('Outside Surface')); await click(find.text('Outside Surface'));
expect(clickedOutside, isEmpty); expect(tappedOutside, isEmpty);
}); });
testWidgets('TapRegionSurface detects inside taps', (WidgetTester tester) async { testWidgets('TapRegionSurface detects inside taps', (WidgetTester tester) async {
final Set<String> clickedInside = <String>{}; final Set<String> tappedInside = <String>{};
await tester.pumpWidget( await tester.pumpWidget(
Directionality( Directionality(
textDirection: TextDirection.ltr, textDirection: TextDirection.ltr,
...@@ -113,21 +113,21 @@ void main() { ...@@ -113,21 +113,21 @@ void main() {
const Text('Outside'), const Text('Outside'),
TapRegion( TapRegion(
onTapInside: (PointerEvent event) { onTapInside: (PointerEvent event) {
clickedInside.add('No Group'); tappedInside.add('No Group');
}, },
child: const Text('No Group'), child: const Text('No Group'),
), ),
TapRegion( TapRegion(
groupId: 1, groupId: 1,
onTapInside: (PointerEvent event) { onTapInside: (PointerEvent event) {
clickedInside.add('Group 1 A'); tappedInside.add('Group 1 A');
}, },
child: const Text('Group 1 A'), child: const Text('Group 1 A'),
), ),
TapRegion( TapRegion(
groupId: 1, groupId: 1,
onTapInside: (PointerEvent event) { onTapInside: (PointerEvent event) {
clickedInside.add('Group 1 B'); tappedInside.add('Group 1 B');
}, },
child: const Text('Group 1 B'), child: const Text('Group 1 B'),
), ),
...@@ -150,39 +150,139 @@ void main() { ...@@ -150,39 +150,139 @@ void main() {
await gesture.removePointer(); await gesture.removePointer();
} }
expect(clickedInside, isEmpty); expect(tappedInside, isEmpty);
await click(find.text('No Group')); await click(find.text('No Group'));
expect( expect(
clickedInside, tappedInside,
unorderedEquals(<String>{ unorderedEquals(<String>{
'No Group', 'No Group',
})); }));
clickedInside.clear(); tappedInside.clear();
await click(find.text('Group 1 A')); await click(find.text('Group 1 A'));
expect( expect(
clickedInside, tappedInside,
equals(<String>{ equals(<String>{
'Group 1 A', 'Group 1 A',
'Group 1 B', 'Group 1 B',
})); }));
clickedInside.clear(); tappedInside.clear();
await click(find.text('Group 1 B')); await click(find.text('Group 1 B'));
expect( expect(
clickedInside, tappedInside,
equals(<String>{ equals(<String>{
'Group 1 A', 'Group 1 A',
'Group 1 B', 'Group 1 B',
})); }));
clickedInside.clear(); tappedInside.clear();
await click(find.text('Outside')); await click(find.text('Outside'));
expect(clickedInside, isEmpty); expect(tappedInside, isEmpty);
clickedInside.clear(); tappedInside.clear();
await click(find.text('Outside Surface')); await click(find.text('Outside Surface'));
expect(clickedInside, isEmpty); expect(tappedInside, isEmpty);
});
testWidgets('Setting the group updates the registration', (WidgetTester tester) async {
final Set<String> tappedOutside = <String>{};
await tester.pumpWidget(
Directionality(
textDirection: TextDirection.ltr,
child: TapRegionSurface(
child: Row(
children: <Widget>[
const Text('Outside'),
TapRegion(
groupId: 1,
onTapOutside: (PointerEvent event) {
tappedOutside.add('Group 1 A');
},
child: const Text('Group 1 A'),
),
TapRegion(
groupId: 1,
onTapOutside: (PointerEvent event) {
tappedOutside.add('Group 1 B');
},
child: const Text('Group 1 B'),
),
],
),
),
),
);
await tester.pump();
Future<void> click(Finder finder) async {
final TestGesture gesture = await tester.startGesture(
tester.getCenter(finder),
kind: PointerDeviceKind.mouse,
);
await gesture.up();
await gesture.removePointer();
}
expect(tappedOutside, isEmpty);
await click(find.text('Group 1 A'));
expect(tappedOutside, isEmpty);
await click(find.text('Group 1 B'));
expect(tappedOutside, isEmpty);
await click(find.text('Outside'));
expect(
tappedOutside,
equals(<String>[
'Group 1 A',
'Group 1 B',
]));
tappedOutside.clear();
// Now change out the groups.
await tester.pumpWidget(
Directionality(
textDirection: TextDirection.ltr,
child: TapRegionSurface(
child: Row(
children: <Widget>[
const Text('Outside'),
TapRegion(
groupId: 1,
onTapOutside: (PointerEvent event) {
tappedOutside.add('Group 1 A');
},
child: const Text('Group 1 A'),
),
TapRegion(
groupId: 2,
onTapOutside: (PointerEvent event) {
tappedOutside.add('Group 2 A');
},
child: const Text('Group 2 A'),
),
],
),
),
),
);
await click(find.text('Group 1 A'));
expect(tappedOutside, equals(<String>['Group 2 A']));
tappedOutside.clear();
await click(find.text('Group 2 A'));
expect(tappedOutside, equals(<String>['Group 1 A']));
tappedOutside.clear();
await click(find.text('Outside'));
expect(
tappedOutside,
equals(<String>[
'Group 1 A',
'Group 2 A',
]));
tappedOutside.clear();
}); });
} }
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