在使用UITextField的过程当中,难免会有限制字符个数,字符输入规则的需求。通常状况下,会有以下两种方法:git
- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string
依然是利用block回调,不过实现方式有点不一样。github
[testField limitCondition:^BOOL(NSString *inputStr){
return ![testField.text isEqualToString:@"111"];
} action:^{
NSLog(@"limit action");
}];
Or
[testField limitNums:3 action:^{
NSLog(@"num limit action");
}];
复制代码
对于UITextField用来作字符限制最好的方法就是使用- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string
这个代理方法,咱们经过判断string来肯定UITextField是否响应输入。缓存
接下来就是如何封装好代理回调的这个过程了 这里我借鉴了 facebook/**KVOController**的思想,建立一个中间管理者,来接管代理方法。 不过须要考虑几个问题:bash
// self 即UITextField 这里这是一个分类方法
self.delegate = UITextFieldDelegateManager.sharedInstance
复制代码
@interface UITextFieldDelegateManager : NSObject<UITextFieldDelegate> {
NSMapTable<id,_LimitInfo *> *_infos;
}
+ (instancetype)sharedInstance;
- (void)addLimitNums:(NSInteger)num key:(id)key target:(id<UITextFieldDelegate>)target action:(void(^)(void))action;
@end
复制代码
@interface _LimitInfo : NSObject
@property(nonatomic,assign)NSInteger num;
@property(nonatomic,weak)id<UITextFieldDelegate> pinocchio;
@end
复制代码
这时,咱们的UITextField的delegate成为了UITextFieldDelegateManager,这样咱们就“截获”了AController的delgate身份。 而这里有一个问题,那就是AController的UITextFieldDelegate内全部方法会失效,这个问题,咱们稍后再说。工具
- (void)addLimitNums:(NSInteger)num key:(id)key target:(id<UITextFieldDelegate>)target action:(void(^)(void))action;
_LimitInfo *info = [_infos objectForKey:key];
if (!info) {
info = [_LimitInfo new];
info.pinocchio = target;
}
info.condition = condition;
[info setConditionAction:action];
[_infos setObject:info forKey:key];
复制代码
这里Key是UITextField当前实例对象,target是AController,咱们把这二者映射进一个NSMapTable中,NSMapTable的弱引用会使咱们不用担忧循环引用。其做用和字典同样。ui
同时,也解决了多个条件约束的问题。atom
而_LimitInfo只是对AController的一个包装,到这时,咱们的AController已经被架空了,成为了一个受咱们摆布的傀儡😏,pinocchio保存了AController的实例。spa
- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string{
BOOL checkInLimit = NO;
_LimitInfo *info = [self safeReadForKey:textField];
if (info.condition && !info.condition(string) && string.length > 0) {
info.conditionAction();
checkInLimit = YES;
}
if (info.num != 0) {
if (info && textField.text.length == info.num && string.length > 0) {
info.action();
checkInLimit = YES;
}
}
if (checkInLimit) {
return NO;
}
if (!info.pinocchio) {
return YES;
}
return [info.pinocchio textField:textField shouldChangeCharactersInRange:range replacementString:string];
}
复制代码
其余方法相似,具体能够参见源码代理
不过这里还要注意咱们刚刚提到的问题,经过咱们的pinocchio return [info.pinocchio textField:textField shouldChangeCharactersInRange:range replacementString:string];
来控制本来逻辑,否则delegate就失效了code
至于代理释放的问题,我是经过runtime hook UITextField的removeFromSuperview方法,在这个方法调用的时候,将pinocchio从新设置回UITextField的delegate,同时移除缓存。
_LimitInfo* info = [_infos objectForKey:key];
((UITextField*)key).delegate = info.pinocchio;
[_infos removeObjectForKey:key];
复制代码