美文网首页ios
IQKeyboardManager 多次调用textFieldS

IQKeyboardManager 多次调用textFieldS

作者: leejnull | 来源:发表于2017-02-23 15:40 被阅读2124次

最新版作者已经解决了这个问题,所以去掉了这个属性isAskingCanBecomeFirstResponder,不需要再判断。下面一大堆可以不用看了~~

------------------------------- 分割线 -------------------------------

IQKeyboardManager是一个非常方便的库,但是使用的时候有一个坑,一直困扰了我很久。
在没使用IQKerboard之前,给一个textField添加点击事件,只需要在

- (BOOL)textFieldShouldBeginEditing:(UITextField *)textField {
       if(textField == myTextField) {
            do something....
            return NO;
       }else {
            return YES;
       }
}

这样就能很简单地做一些操作。
但是在集成了IQKeyboard之后,就会有一个问题,当一个页面有多个textField之后,点击任意一个textfield,IQKeyboard都会强制执行textFieldShouldBeginEditing这个代理方法,而且是多次执行,执行的次数取决于你有多少个textfield添加了delegate。所以就会出现一个很尴尬的情况:
我第二个textfield有一个点击事件,添加代理,放在代理方法内判断执行,但是第一个textfield并没有添加代理,当我点击第一个textfield的时候,就会强制执行shouldBegin这个方法,方法的参数是添加带来的第二个textfield。。。。。。

解决办法想了好几种,一开始是想添加事件

[tf addTarget:self action:@selector(textfieldTapped) forControlEvents:UIControlEventTouchDown];
- (BOOL)textFieldShouldBeginEditing:(UITextField *)textField {
    if(textField == myTextField) {
         return NO;
    }else {
         return YES;
    }
}

但是这样我测试的时候发现当我轻点一下会没有效果,只有重按一下才行,不知道是不是哪里出了问题?
后来又想到

- (void)textFieldDidBeginEditing:(UITextField *)textField {
    if(textField == myTextField) {
         [textField performSelector:@selector(resignFirstResponder) withObject:nil afterDelay:0];
         do something......
    }
}

当进入到这个方法内的时候,键盘已经弹出来了,这时候虽然用这个方法能把键盘再收回去,但屏幕上还是能看到,体验很不好。
解决方案很多,但都不完美。后来在作者的issues里发现作者说了这个

When textField is requested to test that it can become first responder using '[textField canBecomeFirstResponder]; then it asks for it's delegate via 'textFieldShouldBeginEditing' that it can become first responder or not.
I'm asking for 'canBecomeFirstResponder' in 'IQUIView+Hierarchy.m' line no. 118.
I tried by adding controlEvents action on textField, but no luck.
We really need to find out a solution for this case.

command+shift+o进入这个分类里面,看到这个属性

/**
 Returns YES if IQKeyboardManager asking for canBecomeFirstResponder. Useful when doing custom work in textFieldShouldBeginEditing: delegate.
/**
@property (nonatomic, readonly) BOOL isAskingCanBecomeFirstResponder;

好像明白了什么。。。

- (BOOL)textFieldShouldBeginEditing:(UITextField *)textField {
    if (textField == myTextField1) {
        if (textField.isAskingCanBecomeFirstResponder == NO) {
            NSLog(@"do something...");
        }
        return NO;
    }else if (textField == myTextField2) {
        if (textField.isAskingCanBecomeFirstResponder == NO) {
            NSLog(@"do another something...");
        }
        return NO;
    }else {
      return YES;
    }
}

这样就能完美解决每个添加delegate的textfield都会调用textFieldShouldBeginEditing而产生的问题了!!!
事实证明,还是应该多看看作者的源码啊。。。

-------------------------------------分割线--------------------------------------

有人想知道作者这个isAskingCanBecomeFirstResponder是什么?为什么要这么判断,我抽空看了一下作者<IQUIView+Hierarchy.h>这个分类里面的实现,算是大概了解了一下。
关键点在这里,其他的代码都可以不用看,只要看箭头标识的就行

E42F0119-EFAB-40AB-B621-697E0EF5FA41.png

作者通过给分类添加isAskingCanBecomeFirstResponder,来给那些添加了delegate的textField判断是否要执行自定义的事件。

比如说,我这里有5个textfield,我只给tag102、tag103添加了代理,由前面知道,我点击任何一个textfield,都会强制调用textFieldShouldBeginEditing代理方法

4922E95A-2016-4497-9AAC-E3AE532AAE8A.png

当点击没有添加代理的textField的时候,会执行- (NSArray*)responderSiblings这个方法,它会让所有的textfield执行一次-(BOOL)_IQcanBecomeFirstResponder,看第一行代码

[self _setIsAskingCanBecomeFirstResponder:YES];

这个是设置isAskingCanBecomeFirstResponder为YES,通过关联来做到设置属性

-(void)_setIsAskingCanBecomeFirstResponder:(BOOL)isAskingCanBecomeFirstResponder
{
    objc_setAssociatedObject(self, @selector(isAskingCanBecomeFirstResponder), @(isAskingCanBecomeFirstResponder), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

-(BOOL)isAskingCanBecomeFirstResponder
{
    NSNumber *isAskingCanBecomeFirstResponder = objc_getAssociatedObject(self, @selector(isAskingCanBecomeFirstResponder));
    return [isAskingCanBecomeFirstResponder boolValue];
}

重点来了,看第二行代码

BOOL _IQcanBecomeFirstResponder = ([self canBecomeFirstResponder] && [self isUserInteractionEnabled] && ![self isHidden] && [self alpha]!=0.0 && ![self isAlertViewTextField]  && ![self isSearchBarTextField]);

右边调用了一个方法[self canBecomeFirstResponder],如果textField添加了代理,那么调用这个方法就会立即跳转到代理方法内,执行完代理方法才会继续执行后面的内容。

Paste_Image.png

这时候应该很明白了,isAskingCanBecomeFirstResponder被设置为YES了!!!所以判断不通过,返回NO,退出代理方法,跳转回去,继续执行
-(BOOL)_IQcanBecomeFirstResponder后面的内容,同时,在后面它又把这个属性设置成了NO,那么之后再点击tag102的话,直接执行代理方法,就有了我们写的这段代码

if (textField.isAskingCanBecomeFirstResponder == NO) {
            NSLog(@"do something...");
        }

点击未添加代理的textField,会先走作者写的分类中的方法,直接点击添加代理的textField,会只走textField的代理方法,不会走分类的方法了,这时候isAskingCanBecomeFirstResponder默认是NO的,所以重复调用的问题就完全解决了~~~

相关文章

网友评论

  • 言之凿凿:你好,我不明白textfile为什么会有isAskingCanBecomeFirstResponder这个属性,能提示一下吗?非常感谢
    言之凿凿::smiley: thank you
    leejnull:oc category
  • 傅hc:说起来都是泪····搞了一个钟才发现
  • 883c7bb8a6ff:骚年、你这个IQKeyboard是哪个版本,最新的没isAskingCanBecomeFirstResponder这个属性啊
    leejnull:@whyou 很简单,一句代码搞定,textFieldShouldBeginEditing中判断是要弹框的时候,[self.view endEditing:YES];把键盘收回去就行了,然后再弹框,return NO
    883c7bb8a6ff:仔细看了,我的情况和你的不一样,我是每个都有delegate,其中一个是根据情况弹个框,shoubeginediting 返回NO,其余的返回YES,但是点击IQKeyboard的下一个时键盘收不回去,弹框也弹出来了,一直下一个就会一直加弹窗=。=光标一直停留在上一个 遇到过么
    leejnull:@whyou 最新版的作者已经解决这个问题了,所以他把这个属性去掉,不用判断。
  • 4f08d6c62882:太棒了
  • 半夜气笛:只加了一句代码就解决了,wonderful
  • 1f0f77826518:设置完之后 键盘弹出后 视图就不会自动上移了吧
    GPUImage:父控制器和子控制器均在导航栏添加了textFiled, 且都设置了代理, 点击子控制器的textFiled,会调用父控制器的textFiledShouldBeginEditing, 这个应该怎么处理才能不让它调用父控制器的代理方法textFiledShouldBeginEditing?
    leejnull:@riseup 你的问题没明白,这个库是让键盘弹出后,视图自动上移的,而我这个,是为了不让键盘弹出来的。
  • 6aa5635d17c0:isAskingCanBecomeFirstResponder 这个字段什么意思呢 为什么要判断这个啊
    leejnull:我更新了一下,你再看看~
  • Cooperluffy丨路飞: return NO; 键盘不弹出了怎么办?
    Cooperluffy丨路飞:@一个Lee 嗯,我理解错误了,找到方法了,谢谢
    leejnull:就是让键盘不弹出来啊,弹出来你还这样写干什么。。。
  • 落寞King:新手求教.这句代码是什么意思.起什么作用了.
    落寞King:@一个Lee 非常感谢.很适合像我这样的小白理解.
    leejnull:我更新了一下,你看看~ 希望有帮助
    leejnull:源代码我没怎么细看,这个要明白作者这个三方的实现思路~~~
  • 推遍天下无敌手:一万个👍,兄弟
  • Jeremy_Mi:谢谢,正好解决我的问题!
  • 风吹柳絮如花落:给你一亿个赞!!!👍👍👍
  • superGM:正是遇到这个问题,没加上extField.isAskingCanBecomeFirstResponder的判断,只要代理方法有一个是返回yes,所有输入框都会调用-(BOOL)textFieldShouldBeginEditing:(UITextField *)textField
    一开始认为是苹果api的bug
    Cooperluffy丨路飞:返回NO, 我的键盘弹不出来了
    leejnull:这个也是折腾了好久:smile:
  • 冷月天涯:非常棒

本文标题:IQKeyboardManager 多次调用textFieldS

本文链接:https://www.haomeiwen.com/subject/orrbwttx.html