最新版作者已经解决了这个问题,所以去掉了这个属性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>这个分类里面的实现,算是大概了解了一下。
关键点在这里,其他的代码都可以不用看,只要看箭头标识的就行
作者通过给分类添加isAskingCanBecomeFirstResponder,来给那些添加了delegate的textField判断是否要执行自定义的事件。
比如说,我这里有5个textfield,我只给tag102、tag103添加了代理,由前面知道,我点击任何一个textfield,都会强制调用textFieldShouldBeginEditing代理方法
当点击没有添加代理的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添加了代理,那么调用这个方法就会立即跳转到代理方法内,执行完代理方法才会继续执行后面的内容。
这时候应该很明白了,isAskingCanBecomeFirstResponder被设置为YES了!!!所以判断不通过,返回NO,退出代理方法,跳转回去,继续执行
-(BOOL)_IQcanBecomeFirstResponder后面的内容,同时,在后面它又把这个属性设置成了NO,那么之后再点击tag102的话,直接执行代理方法,就有了我们写的这段代码
if (textField.isAskingCanBecomeFirstResponder == NO) {
NSLog(@"do something...");
}
点击未添加代理的textField,会先走作者写的分类中的方法,直接点击添加代理的textField,会只走textField的代理方法,不会走分类的方法了,这时候isAskingCanBecomeFirstResponder默认是NO的,所以重复调用的问题就完全解决了~~~