不用私有API,一行代码获取当前响应链的First Responder

前言

在iOS中,当发生事件响应时,必须知道由谁来响应事件。而UIResponder类就是专门用来响应用户的操作,处理各种事件的,包括触摸事件(Touch Events)、运动事件(Motion Events)和远程控制事件(Remote Control Events)。iOS处理事件的流程将遵循一个不同对象组成的层次结构,也就是响应者链(Responder Chain),网上目前有很多关于响应者链的介绍,这里就不再细讲。在响应者链中非常重要的一个概念就是第一响应者(First Responder),当前第一响应者负责响应事件,或将事件传递给下一响应者。

在编写iOS程序时,我们经常会遇到需要获取当前的第一响应者,例如系统弹出键盘时,我们希望得到当前输入框(也就是第一响应者)的Frame,从而调整视图避免键盘遮挡输入框。然而UIKit并没有提供官方的API专门用于该用途。本文将介绍一种非常简单的且未用到私有API的方法来获取当前第一响应者。

实现思路

常规思路

通过遍历当前UIWindow的所有子视图,从而找到当前的第一响应者。这种方法首先需要做非常多的递归调用,从而判断所有子视图,同时当前响应链上的第一响应者还有可能是子视图的ViewController,这种方法也会漏掉。

使用私有API的思路

使用苹果的私有API可以很容易地解决这个问题,然而苹果不会允许使用私有API的App上架App Store,而且私有API很有可能随时变化,所以这种方式也很不完美。

UIWindow *keyWindow = [[UIApplication sharedApplication] keyWindow];
UIView *firstResponder = [keyWindow performSelector:@selector(firstResponder)];

本文的思路

本文的思路用到的核心Public API就是

- (BOOL)sendAction:(SEL)action to:(id)target from:(id)sender forEvent:(UIEvent *)event

苹果文档对该API的target参数的描述如下:

The object to receive the action message. If target is nil, the app sends the message to the first responder, from whence it progresses up the responder chain until it is handled.

从而可知,利用该API,只要将传入的target设置为nil,则系统会自动顺着响应链查找能够响应action的响应者。我们只需让所有UIResponder的子类都响应我们自定义的action,即可知道当前第一响应者是哪个对象。

实现方法

为实现本文的思路,我们需要为UIResponder提供一个Category(objc)或者Extension(swift)。

Objective-C

// UIResponder+WTYFirstResponder.h
#import <UIKit/UIKit.h>

@interface UIResponder (WTYFirstResponder)
//使用时只需要对UIResponder类调用该类方法即可获得当前第一响应者
+ (id)wty_currentFirstResponder;
@end

//  UIResponder+WTYFirstResponder.m
#import "UIResponder+WTYFirstResponder.h"

static __weak id wty_currentFirstResponder;

@implementation UIResponder (WTYFirstResponder)
+ (id)wty_currentFirstResponder {
    wty_currentFirstResponder = nil;
    // 通过将target设置为nil,让系统自动遍历响应链
    // 从而响应链当前第一响应者响应我们自定义的方法
    [[UIApplication sharedApplication] sendAction:@selector(wty_findFirstResponder:) 
                                               to:nil 
                                             from:nil 
                                         forEvent:nil];
    return wty_currentFirstResponder;
}
- (void)wty_findFirstResponder:(id)sender {
    // 第一响应者会响应这个方法,并且将静态变量wty_currentFirstResponder设置为自己
    wty_currentFirstResponder = self;
}
@end

使用方法

#import "UIResponder+WTYFirstResponder.h"

id firstResponder = [UIResponder wty_firstResponder];

Swift

//  UIResponder+WTYFirstResponder.swift

import UIKit

private weak var wty_currentFirstResponder: AnyObject?

extension UIResponder {

    static func wty_firstResponder() -> AnyObject? {
        wty_currentFirstResponder = nil
        // 通过将target设置为nil,让系统自动遍历响应链
        // 从而响应链当前第一响应者响应我们自定义的方法
        UIApplication.shared.sendAction(#selector(wty_findFirstResponder(_:)), to: nil, from: nil, for: nil)
        return wty_currentFirstResponder
    }
    
    func wty_findFirstResponder(_ sender: AnyObject) {
        // 第一响应者会响应这个方法,并且将静态变量wty_currentFirstResponder设置为自己
        wty_currentFirstResponder = self
    }
}

使用方法

firstResponder = UIResponder.wty_firstResponder()

思路衍生

如果只希望让第一响应者取消其第一响应者的状态,则可以做如下操作:

Objective-C

[[UIApplication sharedApplication] sendAction:@selector(resignFirstResponder) to:nil from:nil forEvent:nil];

Swift

 UIApplication.shared.sendAction(#selector(resignFirstResponder), to: nil, from: nil, for: nil)

Github Repo

本文所用的代码可以在我的Github上找到,如果觉得好用的话请并忙点个星星。


本文个人博客地址: http://wty.im/2016/09/22/get-the-first-responder/
Github: https://github.com/wty21cn/

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 一. Hit-Testing 什么是Hit-Testing?对于触摸事件, window首先会尝试将事件交给事件触...
    面糊阅读 4,479评论 0 50
  • 好奇触摸事件是如何从屏幕转移到APP内的?困惑于Cell怎么突然不能点击了?纠结于如何实现这个奇葩响应需求?亦或是...
    Lotheve阅读 58,965评论 51 604
  • 在iOS开发中经常会涉及到触摸事件。本想自己总结一下,但是遇到了这篇文章,感觉总结的已经很到位,特此转载。作者:L...
    WQ_UESTC阅读 11,311评论 4 26
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 175,971评论 25 709
  • 作为一个简书新人,来谈如何上首页似乎有些大言不惭,在这里,我只是想就这段时间看过的首页热门文章,来谈一下自己的小看...
    一之鱼阅读 3,043评论 2 12

友情链接更多精彩内容