最新版作者已經(jīng)解決了這個(gè)問題绊序,所以去掉了這個(gè)屬性isAskingCanBecomeFirstResponder雷厂,不需要再判斷显歧。下面一大堆可以不用看了~~
------------------------------- 分割線 -------------------------------
IQKeyboardManager是一個(gè)非常方便的庫挟秤,但是使用的時(shí)候有一個(gè)坑贰剥,一直困擾了我很久瞻讽。
在沒使用IQKerboard之前鸳吸,給一個(gè)textField添加點(diǎn)擊事件,只需要在
- (BOOL)textFieldShouldBeginEditing:(UITextField *)textField {
if(textField == myTextField) {
do something....
return NO;
}else {
return YES;
}
}
這樣就能很簡(jiǎn)單地做一些操作速勇。
但是在集成了IQKeyboard之后晌砾,就會(huì)有一個(gè)問題,當(dāng)一個(gè)頁面有多個(gè)textField之后烦磁,點(diǎn)擊任意一個(gè)textfield养匈,IQKeyboard都會(huì)強(qiáng)制執(zhí)行textFieldShouldBeginEditing這個(gè)代理方法,而且是多次執(zhí)行都伪,執(zhí)行的次數(shù)取決于你有多少個(gè)textfield添加了delegate呕乎。所以就會(huì)出現(xiàn)一個(gè)很尷尬的情況:
我第二個(gè)textfield有一個(gè)點(diǎn)擊事件,添加代理陨晶,放在代理方法內(nèi)判斷執(zhí)行猬仁,但是第一個(gè)textfield并沒有添加代理,當(dāng)我點(diǎn)擊第一個(gè)textfield的時(shí)候先誉,就會(huì)強(qiáng)制執(zhí)行shouldBegin這個(gè)方法湿刽,方法的參數(shù)是添加帶來的第二個(gè)textfield。褐耳。诈闺。。铃芦。雅镊。
解決辦法想了好幾種,一開始是想添加事件
[tf addTarget:self action:@selector(textfieldTapped) forControlEvents:UIControlEventTouchDown];
- (BOOL)textFieldShouldBeginEditing:(UITextField *)textField {
if(textField == myTextField) {
return NO;
}else {
return YES;
}
}
但是這樣我測(cè)試的時(shí)候發(fā)現(xiàn)當(dāng)我輕點(diǎn)一下會(huì)沒有效果刃滓,只有重按一下才行仁烹,不知道是不是哪里出了問題?
后來又想到
- (void)textFieldDidBeginEditing:(UITextField *)textField {
if(textField == myTextField) {
[textField performSelector:@selector(resignFirstResponder) withObject:nil afterDelay:0];
do something......
}
}
當(dāng)進(jìn)入到這個(gè)方法內(nèi)的時(shí)候注盈,鍵盤已經(jīng)彈出來了晃危,這時(shí)候雖然用這個(gè)方法能把鍵盤再收回去,但屏幕上還是能看到,體驗(yàn)很不好僚饭。
解決方案很多震叮,但都不完美。后來在作者的issues里發(fā)現(xiàn)作者說了這個(gè)
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進(jìn)入這個(gè)分類里面鳍鸵,看到這個(gè)屬性
/**
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;
}
}
這樣就能完美解決每個(gè)添加delegate的textfield都會(huì)調(diào)用textFieldShouldBeginEditing而產(chǎn)生的問題了;髯铩!贪薪!
事實(shí)證明媳禁,還是應(yīng)該多看看作者的源碼啊。画切。竣稽。
-------------------------------------分割線--------------------------------------
有人想知道作者這個(gè)isAskingCanBecomeFirstResponder是什么?為什么要這么判斷霍弹,我抽空看了一下作者<IQUIView+Hierarchy.h>這個(gè)分類里面的實(shí)現(xiàn)毫别,算是大概了解了一下。
關(guān)鍵點(diǎn)在這里典格,其他的代碼都可以不用看岛宦,只要看箭頭標(biāo)識(shí)的就行
作者通過給分類添加isAskingCanBecomeFirstResponder,來給那些添加了delegate的textField判斷是否要執(zhí)行自定義的事件耍缴。
比如說砾肺,我這里有5個(gè)textfield,我只給tag102防嗡、tag103添加了代理债沮,由前面知道,我點(diǎn)擊任何一個(gè)textfield本鸣,都會(huì)強(qiáng)制調(diào)用textFieldShouldBeginEditing代理方法
當(dāng)點(diǎn)擊沒有添加代理的textField的時(shí)候,會(huì)執(zhí)行- (NSArray*)responderSiblings這個(gè)方法硅蹦,它會(huì)讓所有的textfield執(zhí)行一次-(BOOL)_IQcanBecomeFirstResponder荣德,看第一行代碼
[self _setIsAskingCanBecomeFirstResponder:YES];
這個(gè)是設(shè)置isAskingCanBecomeFirstResponder為YES,通過關(guān)聯(lián)來做到設(shè)置屬性
-(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];
}
重點(diǎn)來了童芹,看第二行代碼
BOOL _IQcanBecomeFirstResponder = ([self canBecomeFirstResponder] && [self isUserInteractionEnabled] && ![self isHidden] && [self alpha]!=0.0 && ![self isAlertViewTextField] && ![self isSearchBarTextField]);
右邊調(diào)用了一個(gè)方法[self canBecomeFirstResponder]涮瞻,如果textField添加了代理,那么調(diào)用這個(gè)方法就會(huì)立即跳轉(zhuǎn)到代理方法內(nèi)假褪,執(zhí)行完代理方法才會(huì)繼續(xù)執(zhí)行后面的內(nèi)容署咽。
這時(shí)候應(yīng)該很明白了,isAskingCanBecomeFirstResponder被設(shè)置為YES了!D瘛窒升!所以判斷不通過,返回NO慕匠,退出代理方法饱须,跳轉(zhuǎn)回去,繼續(xù)執(zhí)行
-(BOOL)_IQcanBecomeFirstResponder后面的內(nèi)容台谊,同時(shí)蓉媳,在后面它又把這個(gè)屬性設(shè)置成了NO,那么之后再點(diǎn)擊tag102的話锅铅,直接執(zhí)行代理方法酪呻,就有了我們寫的這段代碼
if (textField.isAskingCanBecomeFirstResponder == NO) {
NSLog(@"do something...");
}
點(diǎn)擊未添加代理的textField,會(huì)先走作者寫的分類中的方法盐须,直接點(diǎn)擊添加代理的textField玩荠,會(huì)只走textField的代理方法,不會(huì)走分類的方法了丰歌,這時(shí)候isAskingCanBecomeFirstResponder默認(rèn)是NO的姨蟋,所以重復(fù)調(diào)用的問題就完全解決了~~~