- The Gestures that Flutter Deep Dive
- By Nash
- The Nuggets translation Project
- Permanent link to this article: github.com/xitu/gold-m…
- Translator: MeFelixWang
- Proofreader: HaoChuan9421
Flutter provides some great pre-built components for handling touch events, such as in InkWell and InkResponse. Wrap your components around them and they respond to touch events. In addition, it adds a Material style splash effect to your components. For example, InkResponse can optionally control the shape and clipping of the splash as it extends from the edge of the component. What’s interesting is that InkWell and InkResponse don’t do any rendering, but instead update the parent Material component. A common example is a picture. If you wrap the image in inkEll, you’ll notice that the ripple is not visible. This is because it is drawn after the image on the Material. To make the Ink splash visible, wrap the Image with ink. Image. While this is useful for most tasks, you should use GestureDetector if you want to capture more events, such as when the user is dragging the screen.
So what is a gesture detector? How does it work?
Simply put, the gesture detector is a stateless component whose constructor parameters can be used for different touch events. It’s worth noting that you can’t use Pan and Scale at the same time, because Scale is a superset of Pan. The GestureDetector is purely for gesture detection and therefore does not give any visual response (there is no Material Ink propagation).
Here is a table showing the different callbacks provided by GestureDetector and a short description of them:
Property/callback | describe |
---|---|
onTapDown |
Triggered every time the user makes contact with the screenOnTapDown . |
onTapUp |
When the user stops touching the screen,onTapUp Is invoked. |
onTap |
When you touch the screen briefly,onTap Be triggered. |
onTapCancel |
When the user touches the screen but doesn’t finishTap , this event is triggered. |
onDoubleTap |
Called when the screen is touched twice in quick successiononDoubleTap . |
onLongPress |
The user touches the screen over500 millisecondsWhen,onLongPress Be triggered. |
onVerticalDragDown |
When the pointer touches the screen and starts moving vertically,onVerticalDown Is invoked. |
onVerticalDragStart |
When the pointerstartCalled when moving in a vertical directiononVerticalDragStart . |
onVerticalDragUpdate |
This method is called every time the position of the pointer on the screen changes. |
onVerticalDragEnd |
This event is called when the user stops moving and the drag is considered complete. |
onVerticalDragCancel |
Called when the user suddenly stops dragging. |
onHorizontalDragDown |
Called when the user/pointer makes contact with the screen and begins to move horizontally. |
onHorizontalDragStart |
The user/pointer is in contact with the screenstartIt’s moving horizontally. |
onHorizontalDragUpdate |
Called every time the position of the pointer on the horizontal /x axis changes. |
onHorizontalDragEnd |
This event is called at the end of the horizontal drag. |
onHorizontalDragCancel |
When the pointer fails to fireonHorizontalDragDown When the call. |
onPanDown |
Called when the pointer makes contact with the screen. |
onPanStart |
When the pointer event starts moving,onPanStart The trigger. |
onPanUpdate |
Called each time the pointer changes positiononPanUpdate . |
onPanEnd |
After the pan is complete, this event is called. |
onScaleStart |
This event is called when the pointer makes contact with the screen and establishes focus for 1.0. |
onScaleUpdate |
The pointer touching the screen indicates the new focus. |
onScaleEnd |
Called when the pointer is no longer in contact with the screen indicating the end of the gesture. |
The GestureDetector decides which gestures to try to recognize based on which callback is non-empty. This is useful because you need to pass in NULL if you want to disable gestures.
Let’s use the **onTap** gesture as an example to determine how to handle the **GestureDetector**.
First, we create a GestureDetector using the onTap callback. Since it is non-null, the GestureDetector uses our callback when a TAP event occurs. Inside the GestureDetector, you create a Gesture Factory. Gesture Recognizer does a lot of work to determine what Gesture is being processed. This process is the same for all callbacks provided by GestureDetector. The GestureFactories are then passed to the RawGestureDetector.
RawGestureDetector does a lot of work to detect gestures. It is a stateful component that synchronizes all gestures when the state changes, processes the recognizer, gets all pointer events that occur and sends them to the registered recognizer. Then they will battle it out in the gesture arena.
The RawGestureDetectorbuild method consists of a base class Listener that listens for pointer events. This is your preferred class if you want to use raw input from the platform, such as up, down, or cancel events. The Listener doesn’t give you any gestures, just the basic onPointerDown, onPointerUp, onPointerMove and onPointerCancel events. Everything must be handled manually, including reporting yourself to the gesture arena. If you don’t, you won’t get auto-cancel and you won’t be able to participate in the interaction that takes place there. This is the lowest level on the component side.
The Listener is a SingleChildRenderObjectWidget, composed of RenderPointerListener inherited from the RenderProxyBoxWithHitTestBehavior’s class, This means that it mimics the properties of its subclasses while allowing custom HitTestBehavior. If you want to learn more about render boxes and how they work, read this article by Norbert Kozsir.
HitTestBehaviour has three options, deferToChild, Opaque and always. These come from GestureDetector and can be configured in it. DeferToChild passes events down the component tree, which is also the default behavior. Opaque prevents a background component from receiving an event, whereas background component always allows it.
What if you want both parent and child components to receive pointer events?
Let’s imagine for a moment that you have a nested list and you want to scroll through them at the same time. To do this, you need both parent and child components to receive Pointers. You configure the hit test behavior to be semi-transparent, ensuring that both components receive events, but things don’t go according to plan… Why is that?
The answer to the above question is GestureArena.
GestureArena is used for gesture disambiguation. This is where all the recognizers will slug it out and be sent. There can be more than one gesture recognizer at any given point on the screen. The arena takes into account how long the user touches the screen, the slope, and the direction of the drag to determine the winner.
Both parent lists and child lists send their recognizers to the arena, but (as of this writing) only one wins, and it happens to always be the child list.
The fix is to use the GestureFactory with the RawGestureDetector to change the arena’s performance.
As an example, let’s create a simple application that consists of two containers. The goal is for both child and parent containers to receive the gesture.
Wrap both containers with the RawGestureDetector. Next, we will create a custom gesture recognizer AllowMultipleGestureRecognizer. GestureRecognizer is the base class that all other recognizers inherit from. It provides the base API for classes so that they can work/interact with gesture recognizers. It’s worth noting that GestureRecognizer doesn’t care about the nuts and bolts of the recognizer itself.
// Custom gesture recognizer. // Override rejectGesture(). This function is called when a gesture is rejected. By default, it handles // recognizers and cleans them up. But we changed it so that it was actually added manually to be processed instead of the recognizer. The result is that you will have two recognizers to win in the arena. It's a win-win. class AllowMultipleGestureRecognizer extends TapGestureRecognizer { @override void rejectGesture(int pointer) { acceptGesture(pointer); }}Copy the code
In the code above, we are to create a custom class to inherit from TapGestureRecognizer AllowMultipleGestureRecognizer. This means that it can inherit from TapGestureRecognizer. In this example, we have rewritten the rejectGesture so that instead of processing the recognizer, we accept it manually.
Now we pass the custom gesture recognizer GestureRecognizerFactoryWithHandlers RawGestureDetector.
Widget build(BuildContext context) {
returnRawGestureDetector( gestures: { AllowMultipleGestureRecognizer: GestureRecognizerFactoryWithHandlers< AllowMultipleGestureRecognizer>( () => AllowMultipleGestureRecognizer(), / / the constructor (AllowMultipleGestureRecognizer instance) {/ / initializer instance. The onTap = () = >print('Episode 4 is best! (parent container) '); })},Copy the code
Now we pass the custom gesture recognizer GestureRecognizerFactoryWithHandlers RawGestureDetector. The factory function requires two properties, a constructor and an initializer, to construct and initialize the gesture recognizer. We use lambda to pass these arguments. As above mentioned in the code, the constructor returns a new instance of AllowMultipleGestureRecognizer, whereas the initializer get used to listen the tap and some text attribute instance of printed to the console. The two containers will repeat the process, the only difference being the printed text.
Here is the full source code for the sample application:
import 'package:flutter/gestures.dart';
import 'package:flutter/material.dart'; // The main function. The entry of a Flutter application voidmain() { runApp( MaterialApp( home: Scaffold( body: DemoApp(), ), ), ); } // A simple demo application consisting of two containers. The goal is to allow multiple gestures into the arena. // Everything is through 'RawGestureDetector' and custom gesturecognizer (from 'TapGestureRecognizer') ` AllowMultipleGestureRecognizer ` add to the list of gestures, And create a ` AllowMultipleGestureRecognizer ` types of ` GestureRecognizerFactoryWithHandlers `. // It creates a gesture recognizer factory function with the given callback, in this case 'onTap'. // It listens for an instance of 'onTap' and prints text to the console when called. Note that the 'RawGestureDetector' is the same for both containers. The only difference is the printed text (used to identify components). class DemoApp extends StatelessWidget { @override Widget build(BuildContext context) {return RawGestureDetector(
gestures: {
AllowMultipleGestureRecognizer: GestureRecognizerFactoryWithHandlers<
AllowMultipleGestureRecognizer>(
() => AllowMultipleGestureRecognizer(),
(AllowMultipleGestureRecognizer instance) {
instance.onTap = () => print('Episode 4 is best! (parent container) '); },)}, behavior: opaque, // Parent: Container(color: Colors. BlueAccent, child: Center (/ / the two containers with RawGestureDetector wrapped child: RawGestureDetector (gestures: {AllowMultipleGestureRecognizer: GestureRecognizerFactoryWithHandlers< AllowMultipleGestureRecognizer>( () => AllowMultipleGestureRecognizer(), / / the constructor (AllowMultipleGestureRecognizer instance) {/ / initializer instance. The onTap = () = >print('Episode 8 is best! (nested container)'); },)}, // Create a nested container in the first container. Child: Container(color: Colors. YellowAccent, width: 300.0, height: 400.0,),); }} // Custom gesture recognizer. // Override rejectGesture(). This function is called when a gesture is rejected. By default, it handles // recognizers and cleans them up. But we changed it so that it was actually added manually to be processed instead of the recognizer. The result is that you will have two recognizers to win in the arena. It's a win-win. class AllowMultipleGestureRecognizer extends TapGestureRecognizer { @override void rejectGesture(int pointer) { acceptGesture(pointer); }}Copy the code
So what is the result of running the code above?
When you click on the yellow container, both components receive a TAP event, so two statements are printed to the console.
Applications:
Console output:
What happens when you win?
After a gesture wins, the arena is either closed or swept. This will discard unused recognizers and reset the arena. The victory sign then performs the action.
Returning to our Tap example, after this, the function mapped to onTap will now be executed.
conclusion
Today we learned how the Flutter framework handles gestures. We first learned about the dreamy prefabricated components that Flutter provides for processing TAPS and other touch events. Next, we discussed the GestureDetector and experimented with its internal workings. Using an example, we learned how Flutter handles Tap gestures. We walked through the land of the Raw Detector, listened to the Listener and paid tribute to the mysterious Flutter fight club called the GestureArena.
Finally, we introduced most of the gesture systems in Flutter from an application perspective. Armed with this knowledge, you should now have a better understanding of how to capture on-screen touches and process them behind the scenes. If you have any questions or concerns, feel free to comment or contact me via Twitterverse.
Thanks also to Simon Lightfoot (aka “Flutter Whisperer”) for his contribution to this article ❤
- Nash
If you find any mistakes in your translation or other areas that need to be improved, you are welcome to the Nuggets Translation Program to revise and PR your translation, and you can also get the corresponding reward points. The permanent link to this article at the beginning of this article is the MarkDown link to this article on GitHub.
The Nuggets Translation Project is a community that translates quality Internet technical articles from English sharing articles on nuggets. The content covers Android, iOS, front-end, back-end, blockchain, products, design, artificial intelligence and other fields. If you want to see more high-quality translation, please continue to pay attention to the Translation plan of Digging Gold, the official Weibo, Zhihu column.