UITextView限制输入字数(封装控件)

LimitTextView


  整篇文章和代码都基本都是推翻重写,而且可能与网上主流的一些做法不同,我在后面说一下我这么做的一些考虑。当然我还是刚入门不久,肯定写的有不尽人意,如果有更好的方法可以跟我说一下。

  我稍微整合了一下项目里关于这个功能的其他要求,其实也是一些常见的要求,我把他们封装成了一个控件LimitTextView,而且这些附带的功能也是能够自己决定是否开启的。

  下载地址:git
  ps:本文是以中文两个字节,英文一个字节来处理字数限制的。emoji的长度根据emoji不同字节也不同

  可以看看封装好的LimitTextView里都有哪些可以进行操作的属性

         //限制输入的字数(以中文为单位),不赋值就是不限制字数
        @property (assign, nonatomic) NSInteger limteNum; 

       //根据输入文本自适应行高,默认不自适应    
        @property (assign, nonatomic) BOOL autoHeight; 

        //placehold的文字,不设置就不显示       
        @property (strong, nonatomic) NSString *placeHold;

        //placehold的font,如果用到placehold才设置
        //因为placehold自适应行高需要知道字号 
        @property (strong, nonatomic) UIFont *placeHoldFont;

从属性名就能看出LimitTextView具备了哪些功能:

  • textView可以设置限制输入的字数。
  • textView能选择是否够模仿显示出placeHolder(如果你给placeHold赋值即是显示),并且placeHolder会自适应行高。
  • 如果输入的内容超出textView init的时候的frame,会自动增高。(不是从一开始输入多少就自适应多少,而是一旦超过初始化时的高度,就会开始自适应高度!!!)

关键代码

    - (void)textViewDidChange:(UITextView *)textView
    {
    if (self.placeHold.length > 0) {
        //textview没有placeholder,模仿placeholder的状态来控制placeholderLabel的显示和消失
        //textview长度为0
        if (self.text.length==0){
            self.placeHoldLabel.hidden = NO;
        }
        else{
            self.placeHoldLabel.hidden=YES;
        }
    }

    UITextRange *selectedRange = [textView markedTextRange];
    //获取高亮部分
    UITextPosition *pos = [textView positionFromPosition:selectedRange.start offset:0];
    //如果在变化中是高亮部分在变,就不计算字符
    if (selectedRange && pos) {
        return;
    }

    //字数超过后通过退格键把字数删除到符合限制的范围内,会调用这个函数
    //将limitString赋值为@“”,方便下一次循环判断
    if ([textView.text charNumber] <= self.limitNum * 2 && ![self.limitString isEqualToString:@""]) {
        self.limitString = @"";
    }

    //超过规定的字数时
    if ([textView.text charNumber] > self.limitNum * 2) {
        //判断是否是普通的字符或asc码,我就是拿来判断是不是纯英文
        BOOL asc = [textView.text canBeConvertedToEncoding:NSASCIIStringEncoding];
        if (asc){
            //是纯英文就好办,可以直接截取相应的字符数用来显示
            NSString *str = [textView.text substringToIndex:self.limitNum * 2];
            if ([self.limitdelegate respondsToSelector:@selector(beyondLimitNum)]) {
                [self.limitdelegate beyondLimitNum];
            }
            [textView setText:str];
        }else{
            //当前输入的第一次超过才会进行循环截取,如果继续输入直接赋值为截取的符合字数限制的字符串,避免再次循环
            if ([self.limitString isEqualToString:@""]) {
                //如果是中英文混合输入,那么就不能单纯用substringToIndex截取,因为它一视同仁
                //不管中英文都当做一个单位来截断
                __block NSString *str = [[NSString alloc] init];
                __block NSInteger num = 0;
                __weak LimitTextView * bSelf = self;
                //逐字遍历,不管是中文英文,中文就按照字,英文就是按字母
                [textView.text enumerateSubstringsInRange:NSMakeRange(0, [textView.text length])
                                                  options:NSStringEnumerationByComposedCharacterSequences
                                               usingBlock: ^(NSString* substring, NSRange substringRange, NSRange enclosingRange, BOOL* stop) {
                                                   NSInteger subLen = [substring charNumber];
                                                   num += subLen;
                                                   NSLog(@"%d", num);
                                                   if (num <= self.limitNum * 2) {
                                                       str  = [str stringByAppendingString:substring];
                                                   }else{
                                                       if ([bSelf.limitdelegate respondsToSelector:@selector(beyondLimitNum)]) {
                                                           [bSelf.limitdelegate beyondLimitNum];
                                                       }
                                                       self.limitString = str;
                                                       return;
                                                   }
                                               }];

            }
            [textView setText:self.limitString];
        }
    }

    //滚动到一个特定区域
    [self textViewIndicateMoveToCurrentPosition];
    //随着输入越多字数,textview不断变高
    if (self.autoHeight) {
        if (self.contentSize.height > self.frame.size.height){
            CGRect frame = self.frame;
            frame.size.height = self.contentSize.height;
            self.frame = frame;
        }
    }
    }

  这就是所有的关键代码,你肯定很好奇为什么没有- (BOOL)textView:(UITextView *)textView shouldChangeTextInRange:(NSRange)range replacementText:(NSString *)text这个函数

  天朝语言自带神奇的力量,在中文联想输入的时候是不会触发shouldChangeTextInRange:这个函数的。

  是不是想问我神马是联想输入呐?(我是不是很了解不知道的童鞋╮(╯▽╰)╭,因为一开始我也get不到啊(┬_┬))!!简单来说,就是粗线了下面这个图片圈起来的这个地方,而且你选择了这个地方的文字,此时的输入就代表着联想输入。
联想输入图片

说明

  • 关于循环计算字数的代码我没有写到shouldChangeTextInRange:里是因为,textView里面获取的textView.text是已经输入的字 + 高亮的字(不包括当前正在输入的那个高亮拼音/英文)。而我希望的结果是只计算已经确认输入的字的字数。并且,如果我限制输入字数为5个字,我已经输入“I喜欢吃辣”,当你要输入最后一个辣椒的“椒”的时候,刚好输入了拼音“j”,然后此时计算字数,前面中英文一共9个字节加上高亮的“j”,会发现已经到达10个字节的限制,接着如果return NO;你之后的拼音就再也没法输入了,感觉这种处理不是很人性化。
  • 使用循环拼接出于一种考虑,如果我已经输入了四个字符“我爱吃辣”,然后我一次性输入“的东西”三个字,我希望能够自动截取第一个字“的”,舍弃后面两个字“东西”。如果你的处理方式是直接放弃引起输入超过限制时输入的所有字符,可以不需要用到循环。直接用一个变量保存超过限制前的字符串重新赋值给textView.text.
  • 如果一开始我就用中文联想输入,shouldChangeTextInRange:无法调用,所以我把处理placeHold显示/隐藏的逻辑写到textViewDidChange里,避免出现用联想输入而删除placeHold的逻辑没执行引起的覆盖情况。
  • 假设我现在字数限制为5个字,已经输入了“我爱吃辣椒”,但是用户不知道已经到了字数上限决定继续输入,就会走到循环里,然后可能进行多次继续输入的尝试,这时就会多次走入循环。为了避免这种情况,引入了变量self.limitString,来进行判断是否属于上述情况。self.limitString用来保存一个连续超出的输入操作时符合限制字数的string(不知道怎么解释了,看程序就能明白的!)。如果我在超出后(self.limitString不等于@“”)进行删除操作,删除字符到符合要求的范围内就会将self.limitString赋值为@“”,这样下一次输入超过限制字数后会走入循环。用户可能在删除字符后输入不同的内容,所以要重新走一次循环,进行一次字符串拼接。

感觉说明的有点绕,我觉得看程序会配合起来会比较好理解,语死早请原谅我啊T^T,我很尽力在解释了!

关于效率

  肯定是在shouldChangeTextInRange:进行return来限制操作效率高。我这样做效率感觉还是比较低的?感觉循环这种事如果字数上来了,效率肯定会不高。但是我没找到一个比较简单点的符合我要求的方法。如果有大家可以告诉我一下,不胜感激呢!

关于使用

    #import "ViewController.h"
    #import "LimitTextView.h"
    @interface ViewController ()<LimitTextViewDelegate>

    @end

    @implementation ViewController

    - (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];

    LimitTextView *tx = [[LimitTextView alloc]initWithFrame:CGRectMake(0, 60, self.view.frame.size.width, 80)];
    tx.backgroundColor = [UIColor lightGrayColor];
    tx.limteNum = 5;
    tx.autoHeight = YES;
    tx.placeHoldFont = [UIFont systemFontOfSize:13];
    tx.placeHold = @"我是placeholder,我的行高可以改变!!!!";
    tx.limitdelegate = self;
    [self.view addSubview:tx];
    }

    //如果超出了限制字数会走这个delegate,可以把提示的逻辑写到这个里面
    - (void)beyondLimitNum {
        NSLog(@"超过了限制字数( ⊙ o ⊙ )啊!");
    }

推荐看,博主讲的很好,不过博主好像是中英文都当一个字符来处理,没有分开处理,但是文章很不错:

详释(常见UITextView 输入之字数限制)之一---固定长度
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值