UIResponder学习

UIResponder

介绍

UIResponder 这个类定义了很多用来处理响应和时间处理的类。他的子类有UIApplication,UIView以及UIWindow等。

IOS中分为两类事件:触摸事件,和移动事件。最原始的事件处理方是touchesBegan:withEvent:,touchesMoved:withEvent:touchesEnded:withEvent:, and touchesCancelled:withEvent:无论任何时候手指只要触摸屏幕或是在屏幕上移动拖拽甚至离开屏幕都会导致一个UIEvent对象产生。

Responder Chain

在UIResponder中有一个非常重要的概念叫做Responder Chain,个人的理解是这是按照一定规则组织的响应、处理事件的一条链表。在了解UIResponder之前还得在了解一个概念Hit-Testing。在IOS中通常使用hit-testing去找到那个被触摸的视图。这个视图叫hit-test view,当IOS找到hit-test view后就把touch event交个那个视图来处理。下面画个图来说明一下,当点击视图E时看一下hit-testing的工作过程。

1.确定改触摸事件发生在view A范围内,接下来测试view B以及view C。

2.检查发现事件不再view B范围内发生,接下来检查view C发现触摸事件发生在了view C中,所以检查 view D,view E。

3.最后发现事件发生在view E的范围内所以view E成为了hit-test view。

下面是关于调用hit-test的官方说明:

The hitTest:withEvent: method returns the hit test view for a given CGPoint and UIEvent. The hitTest:withEvent: method begins by calling thepointInside:withEvent: method on itself. If the point passed into hitTest:withEvent: is inside the bounds of the view, pointInside:withEvent: returns YES. Then, the method recursively calls hitTest:withEvent: on every subview that returns YES.

Responder Chain 是由responder对象组成的

responder chain是由一系列responder对象连接起来的,他从第一个responder对象开始一直到application对象结束。如果第一个responder不能够处理该事件则该事件会被发送到下一个在该responder chain中的responder来处理。
当自己定义的一个类想让他成为first responder时需要做两件事:
1.重写 canBecomeFirstResponder 方法让他返回YES
2.接受 becomeFirstResponder 消息,如果必要的话可让对象给自己发送该消息。

在这里有一个地方需要注意,当把一个对象变为first responder是要确保这个对象的图形界面已经建立起来,也就是说要在viewDidAppear中调用becomeFirstResponder方法。如果在veiwWillAppear方法中调用becomeFirstResponder将会得到一个NO。

Responder Chain 遵循一个特定的传播路径

如果hit-test view不能够处理该事件则UIKit会将事件传递给下一个Responder。下图则显示了事件在Responder Chain中传播的两种方式:
对于左边的app中事件传播路径如下:
1.初始的界面尝试去处理事件后者消息,打他处理不了则把事件交给它上一层视图处理,因为最开始的界面在他的view controller里的视图层次里不是最上层的。(这里的上下是按照树的结构而言的,下图解释:)
Art/view_hierarchy_relationships.jpg

2.上层视图尝试处理事件,如果他不能处理则将事件交给他的上层视图处理,原因同上。
3.在view controller中最上层的视图尝试处理,他也不能处理则交给他的view controller来处理。
4.view controller也无法处理则交给window来处理。
5.window无法处理交给app object来处理
6.app object无法处理则将该事件丢弃掉。
右边的传播方式稍有不同:
1.一个视图在他的view controller 的视图层中向上传播一个事件直到它到达最顶层视图。
2.最顶层视图无法处理将event交给他的view controller来处理。
3.view controller 传递事件到他的最顶层视图的上一层视图,接下来重复1-3的步骤直到事件到达root view controller。
4.root view controller将事件传递到window object。
5.window 将事件传递给app object。

注意:事件,消息不要自己向上传送而要调用父类中的方法来处理,让UIKit来处理消息在responder chain中的传递。

Responder chain Demo:
/*PMBViewController*/

#import "PMBViewController.h"
#import "PMBViewOne.h"
#import "PMBVIewTwo.h"
#import "PMBVIewThree.h"

@interface PMBViewController ()
@property (weak, nonatomic) IBOutlet PMBVIewThree *viewThree;

@property (weak, nonatomic) IBOutlet PMBVIewTwo *viewTwo;

@property (weak, nonatomic) IBOutlet PMBViewOne *viewOne;


@end

@implementation PMBViewController

- (void)viewDidLoad
{
    [super viewDidLoad];
	// Do any additional setup after loading the view, typically from a nib.
}


- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}
@end

/*PMBViewOne*/
#import "PMBViewOne.h"

@implementation PMBViewOne
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event{
    [super touchesBegan:touches withEvent:event];
    [[[touches anyObject] view] setBackgroundColor:[UIColor redColor]];
}
@end

/*PMBVIewTwo*/
#import "PMBVIewTwo.h"

@implementation PMBVIewTwo
@end

/*PMBVIewThree*/
#import "PMBVIewThree.h"

@implementation PMBVIewThree

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event{
    [super touchesBegan:touches withEvent:event];
    [[[touches anyObject] view] setBackgroundColor:[UIColor greenColor]];
}

@end



  • 3
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
在 iOS 中,可以通过监听键盘弹出事件来实现禁止页面整体被顶上去的功能。具体实现如下: 1. 注册键盘弹出和隐藏的通知,可以在viewDidLoad方法中添加以下代码: ``` NotificationCenter.default.addObserver(self, selector: #selector(keyboardWillShow(_:)), name: UIResponder.keyboardWillShowNotification, object: nil) NotificationCenter.default.addObserver(self, selector: #selector(keyboardWillHide(_:)), name: UIResponder.keyboardWillHideNotification, object: nil) ``` 2. 在键盘弹出和隐藏的方法中,根据键盘的高度来调整页面的布局,可以在UIViewController中添加以下代码: ``` @objc func keyboardWillShow(_ notification: Notification) { if let userInfo = notification.userInfo, let keyboardFrame = userInfo[UIResponder.keyboardFrameEndUserInfoKey] as? NSValue { let keyboardHeight = keyboardFrame.cgRectValue.height // 根据键盘的高度来调整页面的布局 // 比如可以将底部的按钮向上移动键盘的高度 // 这样页面整体就不会被顶上去了 } } @objc func keyboardWillHide(_ notification: Notification) { // 在键盘隐藏时,可以将页面恢复到原来的布局 } ``` 在这两个方法中,可以根据键盘的高度来调整页面的布局,比如将底部的按钮向上移动键盘的高度,这样页面整体就不会被顶上去了。 另外,需要注意的是,在UIViewController中要记得在deinit方法中移除通知的监听,可以添加以下代码: ``` deinit { NotificationCenter.default.removeObserver(self) } ``` 这样就可以实现在iOS中禁止页面整体被顶上去的功能了。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值