Who responds to events
In UIKit we use a responder object (UIResponder) to receive and handle events
A responder object is typically an instance of the UIResponder class, whose subclasses include UIView, UIViewController, and UIApplication
Therefore, we can know that the controls we use daily are responders, such as UIButton, UILabel and so on
In UIResponder and its subclasses, we handle and pass events (UIEvent) through touch (UITouch) methods, including the following methods
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;
- (void)touchesMoved:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;
- (void)touchesEnded:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;
- (void)touchesCancelled:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;
Copy the code
Note that the UITouch parameter passes a collection of UITouch types (not a single UITouch), which corresponds to two or more fingers touching a view at the same time, and one finger for one UITouch
Identify the first responder
Identify the first responder: Find out which control the user most wants to respond to in the touch event
UIApplication fires – (void)sendEvent:(UIEvent *)event after a touch occurs; Pass an encapsulated UIEvent to UIWindow, usually to the currently displayed UIViewController, to the root view of UIViewController
UIKit provides hit-testing to determine responders to touch events
- Unable to receive events
- Do not accept interaction:
view.userInteractionEnabled = false
- Transparency:
The alpha < = 0.01
- Hidden:
view.hidden = true
- Check if coordinates are used internally
pointInside
Method that can be overridden - Traversing the subviews from back to front, FILO, ensures that the last view added is tested first
- If no unchecked sibling views are found when the sibling views are viewed sequentially, the system returns that no sub-views meet the requirements
- (nullable UIView *)hitTest:(CGPoint)point withEvent:(nullable UIEvent *)event;
// recursively calls -pointInside:withEvent:. point is in the receiver's coordinate system
- (BOOL)pointInside:(CGPoint)point withEvent:(nullable UIEvent *)event;
// default returns YES if point is in bounds
Copy the code
For example,
The gray view A in the figure below can be regarded as the root view of the current UIViewController, and the hierarchy of each view is shown on the right. The user’s touch point on the screen is 🌟, and these five views can receive events normally. ⚠️ And note that D is added to A much later than B.
The specific process is as follows:
- We first run A hit test on A, checking according to the process: if 🌟 is inside A (obviously it is), and if A has A subview
- A has two subviews B and D. The subviews are traversed according to the FILO principle. Therefore, A matching test is performed on D and then on B
- Hit test for D: 🌟 is not inside D, indicating that D and its subviews are not the first responders
- B is then hit tested: if 🌟 is inside B (in), and if B has subviews
- B has a subview C, so C needs to be hit again
- Run a hit test on C: 🌟 is not inside C, indicating that C and its subviews are not the first responders
- So you get, the touch point is inside B, but it’s not inside any subview of B
- So B is the first responder, and the hit test ends
- Match test flow: A✅ –> D❎ –> B✅ –> C❎ ❎B
class HitTestExampleView: UIView { override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? { if ! IsUserInteractionEnabled | | isHidden | | alpha < = 0.01 {return nil / /} here refers to the view cannot accept events if the self. The point (inside: point, with: {// Determine whether the touch point is inside the subview in subviews.reversed() {// FILO over the subview let convertedPoint = subview.convert(point, from: self) let resultView = subview.hitTest(convertedPoint, with: If the touch point is inside the child view, return nil if the touch point is not inside the child view. If the touch point is inside the child view, return nil if the touch point is not inside the child view. Return nil {return resultView}} return self // where all the sub-views of the view do not meet the requirements and the touch point is inside the view itself} return nil where the touch point is not inside the view}}Copy the code
Ask a question
There are two views A and B, A is added first, B is added later, but when I click on the screen, I want A to be the first responder, what should I do?
- You can do this by modifying the HIT test method of view B, for example
! isUserInteractionEnabled
Modified toisUserInteractionEnabled
So it’s true and returns nil
if ! IsUserInteractionEnabled | | isHidden | | alpha < = 0.01 {return nil / /} here refers to the view cannot accept eventsCopy the code
Be careful crossing the line
For this example we can see that D has subview E, the process of the hit test.
- Hit test against D: 🌟 is not inside D, so D and its subviews are not first responders
- A match test is performed on B: then a match test is performed on B: 🌟 is inside B (in), and B has subviews
- B has a subview C, so C needs to be hit again
- Hit test on C: 🌟 is not inside C, indicating that C and its subviews are not the first responder and therefore can be obtained. Touch points are inside B, but not in any subviews of B
Finally, the first responder is still B, and even the trend of the whole hit test is the same as before: A✅ –> D❎ –> B✅ –> C❎ ❎ B
The lesson of this example is to be careful if clickable child views are outside the scope of the parent view
– (BOOL)pointInside:(CGPoint)point withEvent:(nullable UIEvent *)event; Method to expand the effective range of click
Events are passed through the response chain
Identify the response chain members
After the first responder is found, the entire response chain is determined
Response chain: A linked list of responders with the first responder at the head of the list
The response chain is essentially the path that passes through a hit test. In the example of the previous chapter, the trend of the whole hit test is as follows: A✅ -> D❎ -> B✅ -> C❎. We remove the left ❎ and connect the first responder B as the head, and the response chain is as follows: B -> A. (There is actually A controller after A, but the controller is not shown in this example, so I will write A)
By default,.next is the parent view of the object if it is of UIView type, with a few exceptions:
- If it’s the root view of UIViewController, then the next responder is UIViewController
- If it’s a UIViewController
- If the root view of UIViewController is the root view of UIWindow, then the next responder is UIWindow object
- If UIViewController is presented by another UIViewController, then the next responder is the second UIViewController
- The next responder to UIWindow is UIApplication
- The next responder to UIApplication is an AppDelegate
For example,
Here’s an example to illustrate. As shown in the figure below, the touch point is 🌟, and according to the hit test, B becomes the first responder. Since C is the superview of B, A is the superview of C, and A is the root view of the Controller, then by rule the response chain looks like this:
View B -> C -> root view A -> UIViewController object -> UIWindow object -> UIApplication object -> App DelegateCopy the code
! [image-20210318205712058](/Users/juice/Library/Application Support/typora-user-images/image-20210318205712058.png)
Passing events along the response chain
The touch event will be first responded by the first responder, triggering its touchesBegan and other methods
If the first responder does not handle the event in the method, the method is passed on to the next responder in the response chain, and so on
If there is no response, it is discarded
For example,
In the picture below, A, B and C are UIViews. We move our finger A certain distance on the screen according to the position of 🌟 and the direction of the arrow, and then release the hand. We should be able to see the output of the lower right image on the console. We can see that view A, VIEW B and view C all actively respond to each event. After each touch occurs, the response method of VIEW B will be triggered first, and then passed to VIEW C and then to view A. But this “positive” response actually means that in our example, neither A, B, or C are suitable recipients of the touch event. They “actively” pass on the event because they look at the information about the event and decide that they are not the right person to handle the event. (Of course, we have three UIViews here, and they really shouldn’t be able to handle events themselves.)
! [image-20210318210153196](/Users/juice/Library/Application Support/typora-user-images/image-20210318210153196.png)
If we replace C with a UIControl class, print the following. We see that the response chain stops at C, and that A’s Touches method is not triggered
This means that in the response chain, UIControl and its subclasses by default don’t pass events on, so that when a control accepts an event, the event is stopped. (UISCrollView also works this way.)
! [image-20210318210231844](/Users/juice/Library/Application Support/typora-user-images/image-20210318210231844.png)
conclusion
In general, the transmission of behind-the-scenes events on the touch screen can be divided into the following four steps:
- Use a hit test to find the first responder
- The response chain is determined by the first responder
- Pass events along the response chain
- The event is either received by a responder or not received by any responder and discarded
Reference blog: juejin.cn/post/689451…
Response chain and gesture recognition
When gesture recognition participates in the response chain
In this case, without UIGestureRecognizer (blue), how does event handling and receiving work with UIGestureRecognizer involved
! [image-20210318213908090](/Users/juice/Library/Application Support/typora-user-images/image-20210318213908090.png)
After finding the first responder through a hit test, the UITouch is distributed to UIResponder’s Touches method, as well as to the gesture recognition system, allowing the two processing systems to work simultaneously
Note that the flow in blue doesn’t just happen once. For example, when you slowly slide a finger across the view, a UITouch object is created. The UITouch updates itself as you move your finger, and also initiates various Touches, such as this trigger sequence
TouchesBegan // Touch the screen touchesMoved // Move your finger on the screen touchesMoved // . touchesMoved // ... TouchesMoved // Move your finger on the screen touchesEnded // Move your finger off the screenCopy the code
The gestureRecognizer property of the UITouch stores gesturecognizer gestures collected during the search for the first user, and the gesture recognition system constantly determines whether the current UITouch matches any of the gesturecognizer gestures collected during the creation of touches
When gesture recognition is successful: The first responder will receive ‘touchesCancelled’ and the view will no longer receive Touches from the UITouch, while other gestures associated with the UITouch will be ‘touchesCancelled’. And it no longer receives touches from the UITouch, so that the gesture it recognizes can monopolize the UITouch
TouchesBegan // Touch the screen touchesMoved // Move your finger on the screen touchesMoved // . touchesMoved // ... 'touchesMoved' // Move finger on screen 'touchesCancelled' // Gesture recognition successful, The Touches method is blocked // now 💅 doesn't leave the screen // But continuing to swipe 🛹 doesn't trigger the Touches methodCopy the code
When the gesture recognition fails: the failure of recognition does not mean that the recognition will not succeed in the future, so the response chain will not be blocked. The internal state of the gesture is, for the most part,.possible, which means that UITouch doesn’t match it for the time being, but it may later identify success, and.fail is really identifying failure
For example,
Now touch and slide a distance on a view with one finger
-
Without gestures, the responder’s ‘TouchBegan’ would be triggered when a finger was pressed down, ‘TouchMoved’ would be triggered when a finger moved, ‘TouchEnded’ would be triggered when a finger lifted, and all the while we were receiving an updated UITouch
-
In the case of adding gestures to the view, we have a response chain working at the same time, and the first half of the gesture recognition system is also recognized. The gesture recognition system determines that the UITouch is acting in accordance with the UIPanGestureRecognizer by sliding the finger a certain distance. The View is sent a ‘touchesCancelled’ message to prevent the UITouch from continuing to trigger the view’s Touches methods. After that, only the target-action method associated with the gesture (dark green node) is called
Further understanding
- The gesture recognizer isn’t a responder, but it also has the Touches method, which fires a bit earlier than the View’s Touches method it adds
- The state of the gesture recognizer is not marked in the figure:
- Gestures are in the picture
recognizing
And at the orange nodes ofrecognized
The brown nodes are all in.possible
state - The state change of the gesture at the green node in the figure is
.began
->[.changed]
->ended
- Gestures are in the picture
More choice
cancelsTouchesInView
: The default is true; If set to false, ‘touchesCancelled’ will not be sent to the view when the gesture is successfully cancelled, i.e. the view’s own methods will not be interrupted, and the final result will be both the gesture and its own method firingdelaysTouchesBegan
: This property defaults tofalse
; If set to true, the View’s Touches method will be delayed until the gesture’s creation succeeds or fails.delaysTouchesEnded
: This property defaults totrue
. The property oftrue
View whentouchesEnded
Will delay firing for approximately 0.15s. This property is often used for combos. If set to false, the view’s touchesEnded methods will not be delayed, they will fire immediately, and our double click will be recognized as two clicks
UIControl with gesture recognition
For user-defined UIControl, gesture recognition takes precedence over event processing by UIControl itself.
For example, when we add a.touchupinside method to a UIControl, we add a UITapGestureRecognizer. Click on the UIControl, and you’ll see that the method associated with the gesture fires and sends touchCancelled to the UIControl, causing its own processing time mechanism to be interrupted and not firing the.touchupinside method.
This leads to a problem: When we add a UIControl subview to a view that already has a click gesture, no matter how much we add a target-action method of click type to UIControl, we end up triggering the parent view’s gesture and breaking UIControl’s processing, Causes the target-action method to never fire
APPLE solution: UIKit makes some controls special. When there is a gesture in the parent view of the control that conflicts with the function of the control, it will trigger its own method first, and not trigger the gesture in the parent view
What if you just want to trigger a gesture? You should add the gesture to the target control instead of the view (as in the example above where UIControl adds a.touchupInside method), and the gesture is in effect