An article to thoroughly understand the Android event distribution mechanism

An article to thoroughly understand the Android event distribution mechanism

Preface

In Android development, we often encounter sliding conflicts (such as the nesting of ScrollView or SliddingMenu and ListView), which require us to have a deep understanding of the Android event response mechanism to solve. The event response mechanism is already essential knowledge for Android developers. It is also a question that interviewers often ask when looking for a job.

Common methods involved in incident response

When the user's finger is in contact with the screen, a series of events are generated through the MotionEvent object, which has four states:

  • MotionEvent.ACTION_DOWN: The moment your finger presses the screen (the beginning of all events)
  • MotionEvent.ACTION_MOVE: The finger moves on the screen
  • MotionEvent.ACTION_UP: The moment your finger leaves the screen
  • MotionEvent.ACTION_CANCEL: Cancel gesture, usually generated by the program, not by the user

The events onClick, onLongClick, onScroll, onFling, etc. in Android are all composed of many Touch events (one ACTION_DOWN, n ACTION_MOVE, 1 ACTION_UP).

The Android event response mechanism is implemented in the form of distribution first (first received by the external View, and then passed to the smallest View in its inner layer in turn) and then processing (starting from the smallest View unit (event source) and passing to the outer layer in turn).

The complexity lies in: the ability to control whether events at each layer continue to be transmitted (distribution and interception are implemented in coordination), as well as the specific consumption of events (event distribution also has event consumption capabilities).

Three important functions involved in android event processing

Event dispatch: public boolean dispatchTouchEvent(MotionEvent ev)

When an event is detected, the Activity first captures it and enters the event dispatch process. (Because the activity does not intercept events, View and ViewGroup do) The event is passed to the dispatchTouchEvent(MotionEvent ev) method of the outermost View, which dispatches the event.

  • return true: Indicates that all events have been digested internally by the View.
  • return false: The event will no longer be distributed at this layer, and will be consumed by the onTouchEvent method of the upper layer control (if the control at this layer is already an Activity, the event will be consumed or processed by the system).
  • If the event dispatch returns the system default super.dispatchTouchEvent(ev), the event will be dispatched to the event interception onInterceptTouchEvent method of this layer for processing.

Event interception: public boolean onInterceptTouchEvent(MotionEvent ev)

  • return true: means intercepting the event and passing it to the onTouchEvent of the control at this layer for processing;
  • return false: means that the event is not intercepted and is successfully distributed to the child View. It is then processed by the child View's dispatchTouchEvent.
  • If super.onInterceptTouchEvent(ev) is returned, it means intercepting the event by default and passing the event to the onTouchEvent method of the current View, which is the same as return true.

Event response: public boolean onTouchEvent(MotionEvent ev)

When dispatchTouchEvent (event dispatch) returns super.dispatchTouchEvent (ev) and onInterceptTouchEvent (event interception returns true or super.onInterceptTouchEvent (ev), the event will be passed to the onTouchEvent method, which responds to the event.

  • If return true, it means that onTouchEvent consumes the event after processing it. At this point, the event ends;
  • If return false, it means no response to the event, then the event will be continuously passed to the onTouchEvent method of the upper View until the onTouchEvent method of a View returns true. If it still returns false when it reaches the top View, then the event is considered not consumed. In the same event series, the current View cannot receive the event again, and the event will be handled by the onTouchEvent of the Activity.
  • If you return super.dispatchTouchEvent(ev), it means that you do not respond to the event, and the result is the same as return false.

From the above process, we can see that no matter dispatchTouchEvent returns true or false, the event will not be distributed. Only when it returns super.dispatchTouchEvent(ev) does it indicate that it has the desire to distribute to the lower layer. However, whether the distribution can be successful needs to be reviewed by the event interception onInterceptTouchEvent. Whether the event is passed up for processing is determined by the return value of onTouchEvent.

View source code analysis

In Android, ImageView, textView, Button, etc. inherit from View but do not override the dispatchTouchEvent method, so they all use this method of View for event distribution.

Look at the source code of some important functions of View:

  1. public boolean dispatchTouchEvent(MotionEvent event) {
  2. //Return true , indicating that the View has digested all events. Return false , indicating that the View has only processed the ACTION_DOWN event, and the event continues to be passed to the parent View (ViewGroup).
  3. if (mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED &&
  4. mOnTouchListener.onTouch(this, event)) {
  5. //The onTouch method here is the callback onTouch() method we rewrite when registering OnTouchListener
  6. return   true ;
  7. } return onTouchEvent(event);
  8. }

First, three conditions are judged:

(1) Check whether the OnTouchListener() event is set for the button;

(2) Whether the control is enabled; (Controls are enabled by default)

(3) Whether the onTouch() method in the OnTouchListener listener implemented in the button returns true;

If all conditions are met, the event is consumed and no longer processed in onTouchEvent. Otherwise, the event will be handed over to the onTouchEvent method for processing.

  1. public boolean onTouchEvent(MotionEvent event) {
  2. ...
  3.  
  4. /* The current onTouch component must be clickable, such as Button, ImageButton, etc. Here, CLICKABLE is true , then it will enter the if method and finally return true .
  5. If it is an ImageView, TextView, or other View that is not clickable by default , CLICKABLE is false and *** returns false . Of course, there are special cases. If an onClick listener is set for these Views , CLICKABLE will also be true here . */
  6. if (((viewFlags & CLICKABLE) == CLICKABLE ||
  7. (viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE)) {
  8. switch (event.getAction()) {
  9. case MotionEvent.ACTION_UP:
  10. ...
  11. if (!post(mPerformClick)) {
  12. performClick(); // Actually, it calls back the onClick() method in the OnClickListener we registered
  13. }
  14. ...
  15. break;
  16. case MotionEvent.ACTION_DOWN:
  17. ...
  18. break;
  19. case MotionEvent.ACTION_CANCEL:
  20. ...
  21. break;
  22. case MotionEvent.ACTION_MOVE:
  23. ...
  24. break;
  25. }
  26. return   true ;
  27. }
  28. return   false ;
  29. }
  1. public boolean performClick() {
  2. ...
  3. // if (li != null && li.mOnClickListener != null ) {
  4. ...
  5. li.mOnClickListener.onClick(this);
  6. return   true ;
  7. }
  8. return   false ;
  9. }
  1. public void setOnClickListener(OnClickListener l) { if (!isClickable()) {
  2.  
  3. setClickable( true );
  4.  
  5. }
  6.  
  7. getListenerInfo().mOnClickListener = l;
  8.  
  9. }

Only when we register OnTouchListener

In the onTouch() method

Return false —> execute onTouchEvent method —> cause onClick() callback method to execute

Return true —> onTouchEvent method is not executed —> causing onClick() callback method not to be executed

ViewGroup source code analysis

The five major layout controls in Android, such as LinearLayout, are all inherited from ViewGroup, and ViewGroup itself is inherited from View, so the event handling mechanism of ViewGroup is effective for these controls.

Part of the source code:

  1. public boolean dispatchTouchEvent(MotionEvent ev) {
  2. final int   action = ev.getAction();
  3. final float xf = ev.getX();
  4. final float yf = ev.getY();
  5. final float scrolledXFloat = xf + mScrollX;
  6. final float scrolledYFloat = yf + mScrollY;
  7. final Rect frame = mTempRect;
  8.    
  9. //This value is false by default , and then we can use the requestDisallowInterceptTouchEvent(boolean disallowIntercept) method
  10. //To change the value of disallowIntercept
  11. boolean disallowIntercept = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0;
  12.    
  13. //Here is the processing logic of ACTION_DOWN
  14. if ( action == MotionEvent.ACTION_DOWN) {
  15. // Clear mMotionTarget, set mMotionTarget to null every time ACTION_DOWN is called    
  16. if (mMotionTarget != null ) {
  17. mMotionTarget = null ;
  18. }
  19.    
  20. //disallowIntercept is false by default , just look at the onInterceptTouchEvent() method of ViewGroup
  21. if (disallowIntercept || !onInterceptTouchEvent(ev)) { //*** point
  22. ev.setAction(MotionEvent.ACTION_DOWN);
  23. final int scrolledXInt = ( int ) scrolledXFloat;
  24. final int scrolledYInt = ( int ) scrolledYFloat;
  25. final View [] children = mChildren;
  26. final int   count = mChildrenCount;
  27. //Traverse its child View    
  28. for ( int i = count - 1; i >= 0; i --) { //The second point  
  29. final View child = children[i];
  30.                       
  31. //If the child View is VISIBLE or the child View is executing animation, it means that the View is
  32. //Can receive Touch events
  33. if ((child.mViewFlags & VISIBILITY_MASK) == VISIBLE
  34. || child.getAnimation() != null ) {
  35. //Get the position range of the child View
  36. child.getHitRect(frame);
  37.                           
  38. //If you touch the point on the screen, it is on the sub -View
  39. if (frame. contains (scrolledXInt, scrolledYInt)) {
  40. // offset the event to the view 's coordinate system
  41. final float xc = scrolledXFloat - child.mLeft;
  42. final float yc = scrolledYFloat - child.mTop;
  43. ev.setLocation(xc, yc);
  44. child.mPrivateFlags &= ~CANCEL_NEXT_UP_EVENT;
  45.                               
  46. //Call the dispatchTouchEvent() method of the child View
  47. if (child.dispatchTouchEvent(ev)) {
  48. // If child.dispatchTouchEvent(ev) returns true , it means
  49. //The event is consumed, set mMotionTarget to the child View    
  50. mMotionTarget = child;
  51. //Return true directly    
  52. return   true ;
  53. }
  54. // The event didn't get handled, try the next   view .
  55. // Don 't reset the event's location, it's not    
  56. // necessary here.
  57. }
  58. }
  59. }
  60. }
  61. }
  62.    
  63. //Judge whether it is ACTION_UP or ACTION_CANCEL
  64. boolean isUpOrCancel = ( action == MotionEvent.ACTION_UP) ||
  65. ( action == MotionEvent.ACTION_CANCEL);
  66.    
  67. if (isUpOrCancel) {
  68. //If it is ACTION_UP or ACTION_CANCEL, set disallowIntercept to the default false    
  69. //If we call the requestDisallowInterceptTouchEvent() method to set disallowIntercept to true    
  70. //When we lift our finger or cancel the Touch event, we need to reset disallowIntercept to false    
  71. //So the above disallowIntercept defaults to false every time we do ACTION_DOWN    
  72. mGroupFlags &= ~FLAG_DISALLOW_INTERCEPT;
  73. }
  74.    
  75. // The event wasn't an ACTION_DOWN, dispatch it to our target if
  76. // we have one.
  77. final View target = mMotionTarget;
  78. //mMotionTarget is null , which means that no View consuming Touch events is found , so we need to call the ViewGroup parent class
  79. //dispatchTouchEvent() method, that is, View 's dispatchTouchEvent() method
  80. if (target == null ) {
  81. // We don 't have a target, this means we' re handling the
  82. // event as a regular view .
  83. ev.setLocation(xf, yf);
  84. if ((mPrivateFlags & CANCEL_NEXT_UP_EVENT) != 0) {
  85. ev.setAction(MotionEvent.ACTION_CANCEL);
  86. mPrivateFlags &= ~CANCEL_NEXT_UP_EVENT;
  87. }
  88. return super.dispatchTouchEvent(ev);
  89. }
  90.    
  91. //The code in this if will not execute ACTION_DOWN, only ACTION_MOVE
  92. // ACTION_UP will go here, if it is intercepted in ACTION_MOVE or ACTION_UP
  93. //Touch event, send ACTION_CANCEL to target, and then return true directly    
  94. //Indicates that this Touch event has been consumed
  95. if (!disallowIntercept && onInterceptTouchEvent(ev)) {
  96. final float xc = scrolledXFloat - ( float ) target.mLeft;
  97. final float yc = scrolledYFloat - ( float ) target.mTop;
  98. mPrivateFlags &= ~CANCEL_NEXT_UP_EVENT;
  99. ev.setAction(MotionEvent.ACTION_CANCEL);
  100. ev.setLocation(xc, yc);
  101.               
  102. if (!target.dispatchTouchEvent(ev)) {
  103. }
  104. // clear the target
  105. mMotionTarget = null ;
  106. // Don't dispatch this event to our own view , because we already
  107. // saw it when intercepting; we just want to give the following
  108. // event to the normal onTouchEvent().
  109. return   true ;
  110. }
  111.    
  112. if (isUpOrCancel) {
  113. mMotionTarget = null ;
  114. }
  115.    
  116. // finally offset the event to the target's coordinate system and    
  117. // dispatch the event.
  118. final float xc = scrolledXFloat - ( float ) target.mLeft;
  119. final float yc = scrolledYFloat - ( float ) target.mTop;
  120. ev.setLocation(xc, yc);
  121.    
  122. if ((target.mPrivateFlags & CANCEL_NEXT_UP_EVENT) != 0) {
  123. ev.setAction(MotionEvent.ACTION_CANCEL);
  124. target.mPrivateFlags &= ~CANCEL_NEXT_UP_EVENT;
  125. mMotionTarget = null ;
  126. }
  127.    
  128. //If ACTION_MOVE and ACTION_DOWN are not intercepted, the Touch event is dispatched directly to the target
  129. return target.dispatchTouchEvent(ev);
  130. }

1. Function of dispatchTouchEvent: Determines whether the event is intercepted and processed by onInterceptTouchEvent.

When super.dispatchTouchEvent is returned, onInterceptTouchEvent determines the direction of the event.

When false is returned, the event will continue to be distributed, and only ACTION_DOWN will be processed internally.

When true is returned, events will not be distributed any more, and all events (ACTION_DOWN, ACTION_MOVE, ACTION_UP) will be processed internally.

2. onInterceptTouchEvent function: intercept events to determine whether the event is passed to the child View

When true is returned, the onTouchEvent is intercepted and handled by itself

When false is returned, the interception is handed over to the child View for processing

3. onTouchEvent function: the event finally reaches this method

When true is returned, all events are processed internally. In other words, subsequent events will continue to be passed to the onTouchEvent() of the view for processing.

When it returns false, the event will be passed upward and accepted by onTouchEvent. If onTouchEvent in the topmost View also returns false, the event will disappear.

Summarize

  • If the ViewGroup finds a View that can handle the event, it will directly hand it over to the child View for processing, and its own onTouchEvent will not be triggered;
  • You can intercept the event of the child View (that is, return true) by overriding the onInterceptTouchEvent(ev) method, and hand the event over to yourself for processing, and your corresponding onTouchEvent method will be executed.
  • The child View can prevent the ViewGroup from intercepting its MOVE or UP events by calling getParent().requestDisallowInterceptTouchEvent(true);
  • After a click event occurs, its transmission process is as follows:
  • Activity->Window->View. ***After View receives the event, it will distribute the event according to the corresponding rules. If a View's onTouchEvent method returns false, it will be handed over to the onTouchEvent method of the parent container for processing, and so on. If all Views do not handle the event, it will be handed over to the onTouchEvent of the Activity for processing.
  • If a View starts processing an event, if it does not consume the ACTION_DOWN event (that is, onTouchEvent returns false), the same event sequence, such as the next ACTION_MOVE, will not be handed over to the View for processing.
  • ViewGroup does not intercept any events by default.
  • Views that are not containers, such as TextView and ImageView, call the onTouchEvent method once they receive an event. They do not have an onInterceptTouchEvent method. Normally, they consume events (return true), unless they are not clickable (both clickable and longClickable are false), in which case they are handled by the onTouchEvent of the parent container.
  • The click event dispatch process is as follows: dispatchTouchEvent—->OnTouchListener's onTouch method—->onTouchEvent-->OnClickListener's onClick method. In other words, the setOnClickListener we usually call has the highest priority, so if onTouchEvent or OnTouchListener's onTouch method returns true, it will not respond to the onClick method.

<<:  How do we do code review?

>>:  A Brief Discussion on iOS Crash (Part 1)

Recommend

Android turns into zombie: Analysis of the new virus "Viking Tribe"

Check Point's research team discovered a new ...

90% of designers agree on these 7 design habits, but they may harm you

When many designers first start designing, they m...

How to operate and promote industrial Internet products?

The operation of industrial Internet products is ...

Miscellaneous: MVC/MVP/MVVM (Part 1)

Preface This article is written to answer a frien...

In 2020, mobile phone manufacturers will not be able to wait for the 5G trend

In the mobile phone industry, the following judgm...

7 tips to quickly improve UI visual experience

Maybe you are a novice designer, or maybe you are...

Why do Toutiao always know what you like to watch?

There are many apps like Toutiao and Qingmang Rea...

Moonlight: Let your phone play PC games

Nowadays, mobile games are becoming more and more...

36 rules for creating hit short videos!

1. Short videos about selling goods 1. Douyin doe...

Build a secure app! iOS security series: HTTPS advanced

[[149556]] This article is divided into the follo...

Building an e-commerce product center from 0 to 1

This series of articles will introduce the produc...