有没有办法让iOS查看哪个孩子有第一响应者身份?

时间:2011-02-17 13:02:18

标签: iphone ipad ios

在Mac OS X中,您可以找到第一个响应者:

[[self window] firstResponder]

在iOS中有什么办法吗?或者您是否需要枚举子控件并向每个控件发送isFirstResponder消息?

4 个答案:

答案 0 :(得分:229)

我非常喜欢VJK's solution,但正如MattDiPasquale所暗示的那样,它似乎比必要的更复杂。所以我写了这个更简单的版本:

目标C

#import "UIResponder+FirstResponder.h"

static __weak id currentFirstResponder;

@implementation UIResponder (FirstResponder)

+(id)currentFirstResponder {
    currentFirstResponder = nil;
    [[UIApplication sharedApplication] sendAction:@selector(findFirstResponder:) to:nil from:nil forEvent:nil];
    return currentFirstResponder;
}

-(void)findFirstResponder:(id)sender {
   currentFirstResponder = self;
}

@end

Swift 4

import UIKit

extension UIResponder {

    private static weak var _currentFirstResponder: UIResponder?

    static var currentFirstResponder: UIResponder? {
        _currentFirstResponder = nil
        UIApplication.shared.sendAction(#selector(UIResponder.findFirstResponder(_:)), to: nil, from: nil, for: nil)

        return _currentFirstResponder
    }

    @objc func findFirstResponder(_ sender: Any) {
        UIResponder._currentFirstResponder = self
    }
}

我也把它作为一种类方法,因为它似乎更有意义。您现在可以找到第一个响应者:[UIResponder currentFirstResponder]


答案 1 :(得分:25)

我在UIResponder上写了一个类别来找到第一个响应者

@interface UIResponder (firstResponder)
- (id) currentFirstResponder;
@end

#import <objc/runtime.h>
#import "UIResponder+firstResponder.h"

static char const * const aKey = "first";

@implementation UIResponder (firstResponder)

- (id) currentFirstResponder {
    [[UIApplication sharedApplication] sendAction:@selector(findFirstResponder:) to:nil from:self forEvent:nil];
    id obj = objc_getAssociatedObject (self, aKey);
    objc_setAssociatedObject (self, aKey, nil, OBJC_ASSOCIATION_ASSIGN);
    return obj;
}

- (void) setCurrentFirstResponder:(id) aResponder {
    objc_setAssociatedObject (self, aKey, aResponder, OBJC_ASSOCIATION_ASSIGN);
}

- (void) findFirstResponder:(id) sender {
    [sender setCurrentFirstResponder:self];
}

@end

然后在任何来自UIResponder的类中,您可以通过调用

获得第一个响应者
UIResponder* aFirstResponder = [self currentFirstResponder];

但请记住首先导入UIResponder类别接口文件!

这使用记录的API,因此不应存在应用商店拒绝问题。

答案 2 :(得分:11)

您需要遍历所有子控件并测试isFirstResponder属性。遇到TRUE时,请退出循环。

UIView *firstResponder;
for (UIView *view in self.view.subviews) //: caused error
{
    if (view.isFirstResponder)
    {
        firstResponder = view;
        break;
    }
}

更好的解决方案

请参阅Jakob's answer

答案 3 :(得分:11)

如果您需要第一响应者,您可以要求它重新签署其状态,这是一种让任何人辞职的方法。 UIView有一个方法可以遍历所有UIViews子视图,并询问任何第一个响应者辞职。

[[self view] endEditing:YES];

这是link to Apple's UIView Docs “此方法查看当前第一个响应者的文本字段的当前视图及其子视图层次结构。如果找到一个,则要求该文本字段作为第一个响应者辞职。如果force参数设置为YES,则为文本领域永远不会被问到;它被迫辞职。“