Unverified Commit 91ab73ab authored by xubaolin's avatar xubaolin Committed by GitHub

Mouse drag should not show selection handles (#69101)

Fixes a bug where selection handles sometimes showed on web.
parent 0297b0b8
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
// found in the LICENSE file. // found in the LICENSE file.
import 'dart:ui' show Offset; import 'dart:ui' show Offset, PointerDeviceKind;
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
...@@ -71,6 +71,7 @@ class DragStartDetails { ...@@ -71,6 +71,7 @@ class DragStartDetails {
this.sourceTimeStamp, this.sourceTimeStamp,
this.globalPosition = Offset.zero, this.globalPosition = Offset.zero,
Offset? localPosition, Offset? localPosition,
this.kind,
}) : assert(globalPosition != null), }) : assert(globalPosition != null),
localPosition = localPosition ?? globalPosition; localPosition = localPosition ?? globalPosition;
...@@ -96,6 +97,9 @@ class DragStartDetails { ...@@ -96,6 +97,9 @@ class DragStartDetails {
/// Defaults to [globalPosition] if not specified in the constructor. /// Defaults to [globalPosition] if not specified in the constructor.
final Offset localPosition; final Offset localPosition;
/// The kind of the device that initiated the event.
final PointerDeviceKind? kind;
// TODO(ianh): Expose the current position, so that you can have a no-jump // TODO(ianh): Expose the current position, so that you can have a no-jump
// drag even when disambiguating (though of course it would lag the finger // drag even when disambiguating (though of course it would lag the finger
// instead). // instead).
......
...@@ -336,7 +336,7 @@ abstract class DragGestureRecognizer extends OneSequenceGestureRecognizer { ...@@ -336,7 +336,7 @@ abstract class DragGestureRecognizer extends OneSequenceGestureRecognizer {
_pendingDragOffset = OffsetPair.zero; _pendingDragOffset = OffsetPair.zero;
_lastPendingEventTimestamp = null; _lastPendingEventTimestamp = null;
_lastTransform = null; _lastTransform = null;
_checkStart(timestamp); _checkStart(timestamp, pointer);
if (localUpdateDelta != Offset.zero && onUpdate != null) { if (localUpdateDelta != Offset.zero && onUpdate != null) {
final Matrix4? localToGlobal = transform != null ? Matrix4.tryInvert(transform) : null; final Matrix4? localToGlobal = transform != null ? Matrix4.tryInvert(transform) : null;
final Offset correctedLocalPosition = _initialPosition.local + localUpdateDelta; final Offset correctedLocalPosition = _initialPosition.local + localUpdateDelta;
...@@ -404,12 +404,13 @@ abstract class DragGestureRecognizer extends OneSequenceGestureRecognizer { ...@@ -404,12 +404,13 @@ abstract class DragGestureRecognizer extends OneSequenceGestureRecognizer {
invokeCallback<void>('onDown', () => onDown!(details)); invokeCallback<void>('onDown', () => onDown!(details));
} }
void _checkStart(Duration timestamp) { void _checkStart(Duration timestamp, int pointer) {
assert(_initialButtons == kPrimaryButton); assert(_initialButtons == kPrimaryButton);
final DragStartDetails details = DragStartDetails( final DragStartDetails details = DragStartDetails(
sourceTimeStamp: timestamp, sourceTimeStamp: timestamp,
globalPosition: _initialPosition.global, globalPosition: _initialPosition.global,
localPosition: _initialPosition.local, localPosition: _initialPosition.local,
kind: getKindForPointer(pointer),
); );
if (onStart != null) if (onStart != null)
invokeCallback<void>('onStart', () => onStart!(details)); invokeCallback<void>('onStart', () => onStart!(details));
......
...@@ -922,8 +922,8 @@ class TextSelectionGestureDetectorBuilder { ...@@ -922,8 +922,8 @@ class TextSelectionGestureDetectorBuilder {
// For backwards-compatibility, we treat a null kind the same as touch. // For backwards-compatibility, we treat a null kind the same as touch.
final PointerDeviceKind? kind = details.kind; final PointerDeviceKind? kind = details.kind;
_shouldShowSelectionToolbar = kind == null _shouldShowSelectionToolbar = kind == null
|| kind == PointerDeviceKind.touch || kind == PointerDeviceKind.touch
|| kind == PointerDeviceKind.stylus; || kind == PointerDeviceKind.stylus;
} }
/// Handler for [TextSelectionGestureDetector.onForcePressStart]. /// Handler for [TextSelectionGestureDetector.onForcePressStart].
...@@ -1077,6 +1077,11 @@ class TextSelectionGestureDetectorBuilder { ...@@ -1077,6 +1077,11 @@ class TextSelectionGestureDetectorBuilder {
/// this callback. /// this callback.
@protected @protected
void onDragSelectionStart(DragStartDetails details) { void onDragSelectionStart(DragStartDetails details) {
final PointerDeviceKind? kind = details.kind;
_shouldShowSelectionToolbar = kind == null
|| kind == PointerDeviceKind.touch
|| kind == PointerDeviceKind.stylus;
renderEditable.selectPositionAt( renderEditable.selectPositionAt(
from: details.globalPosition, from: details.globalPosition,
cause: SelectionChangedCause.drag, cause: SelectionChangedCause.drag,
......
...@@ -557,6 +557,31 @@ void main() { ...@@ -557,6 +557,31 @@ void main() {
expect(renderEditable.selectWordsInRangeCalled, isTrue); expect(renderEditable.selectWordsInRangeCalled, isTrue);
}); });
testWidgets('Mouse drag does not show handles nor toolbar', (WidgetTester tester) async {
// Regressing test for https://github.com/flutter/flutter/issues/69001
await tester.pumpWidget(
const MaterialApp(
home: Scaffold(
body: SelectableText('I love Flutter!'),
),
),
);
final Offset textFieldStart = tester.getTopLeft(find.byType(SelectableText));
final TestGesture gesture = await tester.startGesture(textFieldStart, kind: PointerDeviceKind.mouse);
addTearDown(gesture.removePointer);
await tester.pump();
await gesture.moveTo(textFieldStart + const Offset(50.0, 0));
await tester.pump();
await gesture.up();
await tester.pumpAndSettle();
final EditableTextState editableText = tester.state(find.byType(EditableText));
expect(editableText.selectionOverlay!.handlesAreVisible, isFalse);
expect(editableText.selectionOverlay!.toolbarIsVisible, isFalse);
});
testWidgets('test TextSelectionGestureDetectorBuilder selection disabled', (WidgetTester tester) async { testWidgets('test TextSelectionGestureDetectorBuilder selection disabled', (WidgetTester tester) async {
await pumpTextSelectionGestureDetectorBuilder(tester, selectionEnabled: false); await pumpTextSelectionGestureDetectorBuilder(tester, selectionEnabled: false);
final TestGesture gesture = await tester.startGesture( final TestGesture gesture = await tester.startGesture(
......
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