iOS UITextField输入的中英文长度限制

背景

在用户信息填写等很多场景中,UITextField输入均有中英文字数限制,常见的比如中文100字、英文200字。

方案

方案1

常见的方案如下:

- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string
{
    if (textField == self.titleField) {
        if (string.length == 0) return YES;

        NSInteger existedLength = textField.text.length;
        NSInteger selectedLength = range.length;
        NSInteger replaceLength = string.length;
        if (existedLength - selectedLength + replaceLength > 20){
            return NO;
        }
    }
}

该方法在每输入一个字符时都会调用,在输入拼音联想时,高亮的拼音长度也计算在内,整个字符长度不能超过限制。比如下图中,限制为8个中文(16个英文字符),已输入的中文“测试测试测试”占用12个字符,待输入的“ce s”占用4个字符,至此用户输入的长度已达到输入限制,导致用户之后想输入的拼音都会被截断,最后的一个字很有可能就打不出来了

方案1 示例图

方案2

为了避免方案1存在的问题,我们需要在输入框中存在高亮待输入文字时,不做字长限制判断,输入确认后,再做判断。
1.首先添加监听方法

#pragma mark - 直接添加监听方法
-(void)addTargetMethod{
    [self.textField1 addTarget:self action:@selector(textField1TextChange:) forControlEvents:UIControlEventEditingChanged];
}
-(void)textField1TextChange:(UITextField *)textField{
    NSLog(@"textField1 - 输入框内容改变,当前内容为: %@",textField.text);
}

2.实现输入限制方法,如果高亮联想文字超过限制长度,则截断高亮文字,尽可能保留最长的文字,舍弃多余文字。

方案2 联想输入(字符限制16)

方案2 实际输入(字符限制16)
-(void)textField1TextChange:(UITextField *)textField{
    NSLog(@"textField1 - 输入框内容改变,当前内容为: %@",textField.text);
    NSString *toBeString = textField.text;
    UITextRange *selectedRange = [textField markedTextRange];
    //获取高亮部分
    UITextPosition *position = [textField positionFromPosition:selectedRange.start offset:0];
    // 没有高亮选择的字,则对已输入的文字进行字数统计和限制
    if (!position) {
        textField.text = [toBeString limitedStringForMaxBytesLength:16];
    }
}

这里需要注意的是, 表情符号是个坑。在本文的需求设定中,表情符号占用两个字符长度,相当于一个中文文字。

//ascii算一个 中文算2个 emoji算2个(不标准的做法,根据substringRange可以计算出准确的字节长度)
- (NSString *)limitedStringForMaxBytesLength:(NSUInteger)maxLength {
    __block NSUInteger asciiLength = 0;
    __block NSUInteger subStringRangeLen = 0;
    [self enumerateSubstringsInRange:NSMakeRange(0, self.length)
                             options:NSStringEnumerationByComposedCharacterSequences
                          usingBlock:^(NSString * _Nullable substring, NSRange substringRange, NSRange enclosingRange, BOOL * _Nonnull stop) {
                              unichar uc = [substring characterAtIndex:0];
                              //英文和汉字length都是1
                              if (substringRange.length == 1) {
                                  //这里还有个坑, 有些空格是(uc == 0x2006),不会被 isblank和 isspace命中
                                  //如果不允许出现空格,建议先取出string中的空格
                                  if (isblank(uc) || isspace(uc) || (uc == 0x2006)) {
                                      asciiLength += 1;
                                  } else if (isascii(uc)) {
                                      asciiLength += 1;
                                  } else {
                                      //汉字这里
                                      asciiLength += 2;
                                  }
                              } else {
                                  //表情符号这里
                                  asciiLength += 2;
                              }
                              if (asciiLength <= maxLength) {
                                  subStringRangeLen = substringRange.location + substringRange.length;
                              }
                          }];
    return [self substringWithRange:NSMakeRange(0, subStringRangeLen)];
}
  1. 探索iOS中Emoji表情的编码与解析
  2. iOS UITextField的小技巧——字数限制
  3. UITextField 监听数值变化的三种方法

你可能感兴趣的:(iOS UITextField输入的中英文长度限制)