Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[SuperEditor][Android] Honor handle builders (Resolves #1934) #2272

Merged
merged 5 commits into from
Nov 19, 2024

Conversation

angelosilvestre
Copy link
Collaborator

@angelosilvestre angelosilvestre commented Aug 26, 2024

[SuperEditor][Android] Honor handle builders. Resolves #1934

SuperEditorAndroidControlsController exposes collapsedHandleBuilder and expandedHandlesBuilder, but neither of them are being used anywhere.

This PR changes the android touch interactor to honor the provided builders.

The drag handles involve various gesture callbacks to place/expand/collapse the selection, start/stop the blinking caret, show/hide the toolbar, among other things. It would be very difficult for developers to do that on their own.

Because of that, this PR introduces an object called DocumentHandleGestureCallbacks, which holds the gesture callbacks that should be attached to the drag handle. Developers would wrap the handle with a GestureDetector and set each property to its corresponding GestureDetector callback.

I needed to modify the expanded handles selection logic, because after these changes, when the user starts dragging the expanded handle, the pan update event is trigger before the user moves the drag handle to another character, so we couldn't determine if the user is dragging upstream or downstream. This breaks our logic of selecting by word or character.

The original ticket also mentions that we use magnifierBuilder and toolbarBuilder in different ways. For magnifierBuilder, we expect that developers will include the Follower in the returned widget, while in toolbarBuilder we wrap the returned widget with a Follower. Letting the developers attach the Follower to the tree seems more logical to me, because it would be able to customize the offset and anchors. But changing that now seems like a big breaking change. Should we do that?

@matthew-carroll
Copy link
Contributor

@angelosilvestre is this PR both connecting the existing builders, and also modifying the tooling to build them? If so, can you split this into two PRs: one for fixing the existing bug, and another for whatever improvements you want to make?

@angelosilvestre
Copy link
Collaborator Author

@matthew-carroll I think that at least we should keep DocumentHandleGestureCallbacks in this PR. Otherwise it will be very impractical for anyone to use a custom handle, since the drag handle selection depends on the drag handle gesture callbacks.

@angelosilvestre
Copy link
Collaborator Author

@matthew-carroll Updated removing AndroidDefaultFollowingCollapsedHandle from this PR.

@matthew-carroll
Copy link
Contributor

@angelosilvestre can you update the PR description so that it only describes the changes in this PR after you split this PR into two?

@angelosilvestre
Copy link
Collaborator Author

@matthew-carroll Updated the description.

@matthew-carroll
Copy link
Contributor

@angelosilvestre since you focused in this PR around making it simpler and easier to create custom handles, can you add a code sample to the PR description that shows a minimum setup for a custom drag handle? That will probably help evaluate those decisions.

WRT the Follower issue, let's file a new issue for that and handle it later. We can make the breaking change when it seems like a good time to do that. We've got a lot of stuff in flight right now.

@matthew-carroll
Copy link
Contributor

@angelosilvestre did you update the description? I see that my last comment was the last activity on this thread, so I wasn't sure if that happened, or not.

@angelosilvestre
Copy link
Collaborator Author

@angelosilvestre since you focused in this PR around making it simpler and easier to create custom handles, can you add a code sample to the PR description that shows a minimum setup for a custom drag handle? That will probably help evaluate those decisions.

In this PR I focused on just making the handle builder be honored. Making it simpler and easier would be moved to a separate PR, right?

@matthew-carroll
Copy link
Contributor

I would defer to the earlier conversation in the thread. I think that's all the conversation we've had about this. What I don't know is whether your current PR description describes what's actually in this PR, or whether the description is still about the original PR, which included stuff that you later removed.

@angelosilvestre
Copy link
Collaborator Author

I updated the description.

@matthew-carroll
Copy link
Contributor

I needed to modify the expanded handles selection logic, because after these changes, panning was triggering the pan update while the handle still focused the initially focused character, causing the selection strategy to be confused to determine if the user was dragging upstream or downstream.

I'm not sure what this means. Can you try to describe this in a clearer way?

@angelosilvestre
Copy link
Collaborator Author

Updated it again. Does it look better now?

@matthew-carroll
Copy link
Contributor

Ok, now it reads:

I needed to modify the expanded handles selection logic, because after these changes, when the user starts dragging the expanded handle, the pan update event is trigger before the user moves the drag handle to another character, so we couldn't determine if the user is dragging upstream or downstream. This breaks our logic of selecting by word or character.

Wouldn't it be accurate to say if the user moves left, or up its an upstream selection, and if the user moves right or down then it's a downstream selection? There should probably be a little slop before making that choice, but we shouldn't have to wait for the handle to actually move for us to know the direction of the drag, right?

@angelosilvestre
Copy link
Collaborator Author

Wouldn't it be accurate to say if the user moves left, or up its an upstream selection, and if the user moves right or down then it's a downstream selection? There should probably be a little slop before making that choice, but we shouldn't have to wait for the handle to actually move for us to know the direction of the drag, right?

The change is basically adding that left/right move handling. Do we need to change the whole behavior on this PR?

key: DocumentKeys.upstreamHandle,
handleType: HandleType.upstream,
color: _controlsController!.controlsColor ?? Theme.of(context).primaryColor,
Widget _buildExpandedHandles() {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we need to change this from List<Widget> to Widget? I'm hesitant to add more layout widgets when not needed, because it increases the odds that at some point something isn't aligned correctly because of we've got a Stack in a Stack in a Stack, etc.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We can keep it as a list. Reverted.


return Stack(
children: [
Follower.withOffset(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

When you mentioned in the issue ticket that it's difficult for users to configure their own handles, did you include the Follower in your description of it being difficult? I.e., did you want to wrap the custom handle with a Follower so the user doesn't have to? Or is it better like this to have the user provide the Follower?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we should have the user provide the Follower, to have full control on the alignment, expansion, etc. We could add a default following handle to make it easier for those who don't need this kind of control.

@@ -304,6 +305,91 @@ void main() {
expect(SuperEditorInspector.isCaretVisible(), true);
});

testWidgetsOnAndroid("allows customizing the collapsed handle", (tester) async {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Given that most of this PR is based on delegating gesture behavior, should this PR add a number of tests that use those gestures? Or is that fully covered by existing tests?

Also, why are we only testing for Android?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We currently have test ensuring that dragging the expanded handle changes the selection. Should we add a test for the collapsed handle? We have properties for handle customization on Android only.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Seems like tests for the collapsed handle would be as important as for expanded handles.

@@ -309,6 +309,18 @@ class TestSuperEditorConfigurator {
return this;
}

/// Configures the [SuperEditor] to use the given [builder] as its android collapsed handle builder.
TestSuperEditorConfigurator withAndroidCollapsedHandleBuilder(DocumentCollapsedHandleBuilder? builder) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why only for Android?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The original ticket is only about Android. We don't have handle builders for iOS.

@angelosilvestre angelosilvestre force-pushed the 1934_android-handle-builders-unused branch from 1074a91 to a309fc8 Compare November 13, 2024 19:22
@@ -23,12 +23,24 @@ class DocumentKeys {
/// The [handleKey] is used to find the handle in the widget tree for various purposes,
/// e.g., within tests to verify the presence or absence of the handle.
///
/// The [gestureDelegate] hold event handles that should be attached to the gesture recognizer
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This paragraph is confusing - I'm not sure what exactly it's trying to say.

"The [gestureDelegate] hold event handles". First, it sounds like "hold" -> "holds". Beyond that, I don't know what it means to "hold event handles"...do you mean "handlers"? If so, the delegate doesn't "hold event handlers", a delegate "defines handlers for gesture events that occur on a given handle".

"For example, set [gestureDelegate.onTap] to the handle's onTap event."

I don't know what this is telling me to do. How do I set a method on the handle's onTap? Is this a statement about what I need to do as a reader, or a description of what is happening behind the scenes? I can't tell.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Updated.

/// The [gestureDelegate] hold event handles that should be attached to the gesture recognizer
/// attached to the handle. For example, set [gestureDelegate.onTap] to the handle's `onTap` event.
///
/// Use [shouldShow] to fade in/out the handle entrance/exit, for example, using an [AnimatedOpacity]
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This paragraph is also unclear. When reading this, I can't really tell what you want me to do, or what Super Editor does. I generally understand that something is going on with fading, but there seems to be like 3 distinct details that are smushed into one statement and it's confusing.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I updated trying to make it a bit more clear.

/// These callbacks are intended to make it easier for developers to customize
/// the drag handles, without having to re-implement the gesture logic.
///
/// To use it, for example, wrap the handle in a `GestureDetector` and pass
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you replace this last sentence with a minimal code example showing this in use?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Updated.

@@ -23,12 +23,25 @@ class DocumentKeys {
/// The [handleKey] is used to find the handle in the widget tree for various purposes,
/// e.g., within tests to verify the presence or absence of the handle.
///
/// The [gestureDelegate] defines handlers for gesture events that occur on a given handle. Implementers
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is still unclear as to expectations. It looks like you're trying to describe what implementers need to do. If so, we can be much more direct.

Implementers of this builder have the following responsibilities:
 * Attach the [handleKey] to the widget that renders the handle.
 * Wrap the handle widget with a `Follower` and attach the `focalPoint` to the `Follower`.
 * Wrap the handle widget with a `GestureDetector` and attach the provided `gestureDelegate` callbacks to the `GestureDetector`.
 * When `shouldShow` is false, hide the handle and ensure that no gestures are handled.

Then below that show a code example of one full implementation, which covers all of these bases.

Repeat this approach for the other functions documentation.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Updated.

Copy link
Contributor

@matthew-carroll matthew-carroll left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM

@matthew-carroll matthew-carroll merged commit a3802c0 into main Nov 19, 2024
13 of 14 checks passed
@matthew-carroll matthew-carroll deleted the 1934_android-handle-builders-unused branch November 19, 2024 01:55
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Android handle builders don't seem to be used
2 participants