热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

iOS实现UITextField+Limit的字符限制方法

本文介绍了在iOS开发中使用UITextField实现字符限制的方法,包括利用代理方法和使用BNTextField-Limit库的实现策略。通过这些方法,开发者可以方便地限制UITextField的字符个数和输入规则。

在使用UITextField的过程中,不免会有限制字符个数,字符输入规则的需求。一般情况下,会有如下两种方法:

- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string

BNTextField-Limit 的方法

依然是利用block回调,不过实现方式有点不同。

[testField limitCondition:^BOOL(NSString *inputStr){
        return ![testField.text isEqualToString:@"111"];
    } action:^{
        NSLog(@"limit action");
}];
Or

[testField limitNums:3 action:^{
	NSLog(@"num limit action");
}];

BNTextField-Limit 的实现策略

对于UITextField用来做字符限制最好的方法就是使用 - (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string 这个代理方法,我们通过判断string来确定UITextField是否响应输入。

接下来就是如何封装好代理回调的这个过程了这里我借鉴了 facebook /** KVOController **的思想,创建一个中间管理者,来接管代理方法。 不过需要考虑几个问题:

  • 代理释放的问题
  • 多个条件约束
  • 如何不影响其他代理方法

实现过程:

  1. 首先,我们假定,AController内实现了UITextField的delegate,我们先把delegate的身份接管过来,实现 偷梁换柱
// self 即UITextField 这里这是一个分类方法
  self.delegate =  UITextFieldDelegateManager.sharedInstance
  1. UITextFieldDelegateManager为中间管理类, keyCode:
@interface UITextFieldDelegateManager : NSObject {
    NSMapTable *_infos;
}

+ (instancetype)sharedInstance;

- (void)addLimitNums:(NSInteger)num key:(id)key target:(id)target action:(void(^)(void))action;

@end
@interface _LimitInfo : NSObject

@property(nonatomic,assign)NSInteger num;
@property(nonatomic,weak)id pinocchio;

@end

这时,我们的UITextField的delegate成为了UITextFieldDelegateManager,这样我们就“截获”了AController的delgate身份。 而这里有一个问题,那就是AController的UITextFieldDelegate内所有方法会失效,这个问题,我们稍后再说。

  1. 实现 - (void)addLimitNums:(NSInteger)num key:(id)key target:(id)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的弱引用会使我们不用担心循环引用。其作用和字典一样。

同时,也解决了多个条件约束的问题。

而_LimitInfo只是对AController的一个包装,到这时,我们的AController已经被架空了,成为了一个受我们摆布的傀儡:smirk:,pinocchio保存了AController的实例。

  1. 接下来就简单了,将UITextFieldDelegate在UITextFieldDelegateManager中全部实现出来 主要是我们的shouldChangeCharactersInRange代理方法
- (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就失效了

至于代理释放的问题,我是通过runtime hook UITextField的removeFromSuperview方法,在这个方法调用的时候,将pinocchio重新设置回UITextField的delegate,同时移除缓存。

_LimitInfo* info = [_infos objectForKey:key];
  ((UITextField*)key).delegate = info.pinocchio;
  [_infos removeObjectForKey:key];
  1. 至此,一个基于  facebook / KVOController 思想的小 工具 就出炉了,虽然简单,但是需要这种思想还是比较巧妙的。

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 我们


推荐阅读
author-avatar
花甲墨者
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有