最新版作者已經解決了這個問題,所以去掉了這個屬性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的,所以重復調用的問題就完全解決了~~~