diff --git a/super_editor/lib/src/default_editor/document_gestures_touch_android.dart b/super_editor/lib/src/default_editor/document_gestures_touch_android.dart index f321c17e1..01ef91b0a 100644 --- a/super_editor/lib/src/default_editor/document_gestures_touch_android.dart +++ b/super_editor/lib/src/default_editor/document_gestures_touch_android.dart @@ -31,6 +31,7 @@ import 'package:super_editor/src/infrastructure/platforms/mobile_documents.dart' import 'package:super_editor/src/infrastructure/signal_notifier.dart'; import 'package:super_editor/src/infrastructure/sliver_hybrid_stack.dart'; import 'package:super_editor/src/infrastructure/touch_controls.dart'; +import 'package:super_keyboard/super_keyboard.dart'; import '../infrastructure/document_gestures.dart'; import '../infrastructure/document_gestures_interaction_overrides.dart'; @@ -671,6 +672,9 @@ class _AndroidDocumentTouchInteractorState extends State 0; - } - void _onPanStart(DragStartDetails details) { // Stop waiting for a long-press to start, if a long press isn't already in-progress. _tapDownLongPressTimer?.cancel(); diff --git a/super_editor/lib/src/default_editor/document_gestures_touch_ios.dart b/super_editor/lib/src/default_editor/document_gestures_touch_ios.dart index e18da3a20..ab8686d21 100644 --- a/super_editor/lib/src/default_editor/document_gestures_touch_ios.dart +++ b/super_editor/lib/src/default_editor/document_gestures_touch_ios.dart @@ -31,6 +31,7 @@ import 'package:super_editor/src/infrastructure/platforms/platform.dart'; import 'package:super_editor/src/infrastructure/signal_notifier.dart'; import 'package:super_editor/src/infrastructure/sliver_hybrid_stack.dart'; import 'package:super_editor/src/infrastructure/touch_controls.dart'; +import 'package:super_keyboard/super_keyboard.dart'; import '../infrastructure/document_gestures.dart'; import '../infrastructure/document_gestures_interaction_overrides.dart'; @@ -266,7 +267,6 @@ class IosDocumentTouchInteractor extends StatefulWidget { required this.selection, this.openKeyboardWhenTappingExistingSelection = true, required this.openSoftwareKeyboard, - required this.isImeConnected, required this.scrollController, required this.dragHandleAutoScroller, required this.fillViewport, @@ -289,10 +289,6 @@ class IosDocumentTouchInteractor extends StatefulWidget { /// A callback that should open the software keyboard when invoked. final VoidCallback openSoftwareKeyboard; - /// A [ValueListenable] that should notify this widget when the IME connects - /// and disconnects. - final ValueListenable isImeConnected; - /// Optional list of handlers that respond to taps on content, e.g., opening /// a link when the user taps on text with a link attribution. /// @@ -661,7 +657,7 @@ class _IosDocumentTouchInteractorState extends State selection.extent.nodeId == docPosition.nodeId && selection.extent.nodePosition.isEquivalentTo(docPosition.nodePosition); - if (didTapOnExistingSelection && widget.isImeConnected.value) { + if (didTapOnExistingSelection && SuperKeyboard.instance.state.value == KeyboardState.open) { // Toggle the toolbar display when the user taps on the collapsed caret, // or on top of an existing selection. // diff --git a/super_editor/lib/src/default_editor/super_editor.dart b/super_editor/lib/src/default_editor/super_editor.dart index e18d13806..a37654dfb 100644 --- a/super_editor/lib/src/default_editor/super_editor.dart +++ b/super_editor/lib/src/default_editor/super_editor.dart @@ -919,7 +919,6 @@ class SuperEditorState extends State { selection: editContext.composer.selectionNotifier, openKeyboardWhenTappingExistingSelection: widget.selectionPolicies.openKeyboardWhenTappingExistingSelection, openSoftwareKeyboard: _openSoftwareKeyboard, - isImeConnected: _isImeConnected, contentTapHandlers: [ ..._contentTapHandlers ?? [], for (final plugin in widget.plugins) // diff --git a/super_editor/test/super_editor/mobile/super_editor_android_overlay_controls_test.dart b/super_editor/test/super_editor/mobile/super_editor_android_overlay_controls_test.dart index 19f01956c..65cdd2d02 100644 --- a/super_editor/test/super_editor/mobile/super_editor_android_overlay_controls_test.dart +++ b/super_editor/test/super_editor/mobile/super_editor_android_overlay_controls_test.dart @@ -6,6 +6,7 @@ import 'package:follow_the_leader/follow_the_leader.dart'; import 'package:super_editor/src/infrastructure/platforms/android/selection_handles.dart'; import 'package:super_editor/super_editor.dart'; import 'package:super_editor/super_editor_test.dart'; +import 'package:super_keyboard/test/keyboard_simulator.dart'; import 'package:super_text_layout/super_text_layout.dart'; import '../../test_runners.dart'; @@ -168,6 +169,41 @@ void main() { expect(SuperEditorInspector.isMobileMagnifierVisible(), isFalse); }); + testWidgetsOnAndroid("shows toolbar when long pressing on an empty paragraph and hides it after typing", + (tester) async { + await tester // + .createDocument() + .withSingleEmptyParagraph() + .pump(); + + // The decision about showing the toolbar depends on the keyboard visibility. + // Simulate the keyboard being visible immediately after the IME is connected. + TestSuperKeyboard.install(tester, keyboardAnimationTime: Duration.zero); + addTearDown(() => TestSuperKeyboard.uninstall()); + + // Ensure the toolbar is not visible. + expect(SuperEditorInspector.isMobileToolbarVisible(), isFalse); + + // Long press to show the toolbar. + final gesture = await tester.longPressDownInParagraph('1', 0); + + // Ensure the toolbar is visible. + expect(SuperEditorInspector.isMobileToolbarVisible(), isTrue); + + // Release the finger. + await gesture.up(); + await tester.pump(); + + // Ensure the toolbar is still visible. + expect(SuperEditorInspector.isMobileToolbarVisible(), isTrue); + + // Type a character to hide the toolbar. + await tester.typeImeText('a'); + + // Ensure the toolbar is not visible. + expect(SuperEditorInspector.isMobileToolbarVisible(), isFalse); + }); + testWidgetsOnAndroid("shows magnifier when dragging expanded handle", (tester) async { await _pumpSingleParagraphApp(tester); diff --git a/super_editor/test/super_editor/mobile/super_editor_ios_overlay_controls_test.dart b/super_editor/test/super_editor/mobile/super_editor_ios_overlay_controls_test.dart index 7db3549a2..f16fa62b6 100644 --- a/super_editor/test/super_editor/mobile/super_editor_ios_overlay_controls_test.dart +++ b/super_editor/test/super_editor/mobile/super_editor_ios_overlay_controls_test.dart @@ -4,6 +4,7 @@ import 'package:flutter_test_robots/flutter_test_robots.dart'; import 'package:flutter_test_runners/flutter_test_runners.dart'; import 'package:super_editor/super_editor.dart'; import 'package:super_editor/super_editor_test.dart'; +import 'package:super_keyboard/test/keyboard_simulator.dart'; import 'package:super_text_layout/super_text_layout.dart'; import '../../test_runners.dart'; @@ -132,6 +133,41 @@ void main() { expect(SuperEditorInspector.isMobileMagnifierVisible(), isFalse); }); + testWidgetsOnIos("shows toolbar when long pressing on an empty paragraph and hides it after typing", + (tester) async { + await tester // + .createDocument() + .withSingleEmptyParagraph() + .pump(); + + // The decision about showing the toolbar depends on the keyboard visibility. + // Simulate the keyboard being visible immediately after the IME is connected. + TestSuperKeyboard.install(tester, keyboardAnimationTime: Duration.zero); + addTearDown(() => TestSuperKeyboard.uninstall()); + + // Ensure the toolbar is not visible. + expect(SuperEditorInspector.isMobileToolbarVisible(), isFalse); + + // Long press, this shouldn't show the toolbar. + final gesture = await tester.longPressDownInParagraph('1', 0); + + // Ensure the toolbar is not visible yet. + expect(SuperEditorInspector.isMobileToolbarVisible(), isFalse); + + // Release the finger. + await gesture.up(); + await tester.pump(); + + // Ensure the toolbar is visible. + expect(SuperEditorInspector.isMobileToolbarVisible(), isTrue); + + // Type a character to hide the toolbar. + await tester.typeImeText('a'); + + // Ensure the toolbar is not visible. + expect(SuperEditorInspector.isMobileToolbarVisible(), isFalse); + }); + testWidgetsOnIos("does not show toolbar upon first tap", (tester) async { await tester // .createDocument()