IQKeyboardManager 多次调用textFieldShouldBeginEditing

最新版作者已经解决了这个问题,所以去掉了这个属性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的,所以重复调用的问题就完全解决了~~~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容