iOS 键盘联想特殊字符问题

话不多说先上图

[图片上传失败...(image-3146a2-1657457047127)]

做了个需求,产品要求右边字数限制是字节数限制,中文汉子2字节,英文字母、数字都是1字节。我们用GBK编码来计算字节数。

需求不难,计算字节方法如下:

// 计算字节的个数
- (NSUInteger)convertToByte:(NSString *)str {
    NSStringEncoding encoding = CFStringConvertEncodingToNSStringEncoding(kCFStringEncodingGB_18030_2000);
    NSUInteger strLength = 0;
    char *p = (char *)[str cStringUsingEncoding:encoding];

    NSUInteger lengthOfBytes = [str lengthOfBytesUsingEncoding:encoding];
    for (int i = 0; i < lengthOfBytes; i++) {
        if (*p) {
            p++;
            strLength++;
        } else {
            p++;
        }
    }
    return strLength;
}

实现这个功能也很简单,只需要在UITextField或者UITextView代理方法中设置即可。

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

{
    NSString *newStr = [textField.text stringByReplacingCharactersInRange:range withString:string];
    if ([self convertToByte:newStr] > 30) {
        return NO;
    }
    self.remarkNameLabel.text = [NSString stringWithFormat:@"%@ %@", [JWDataHelper sharedDataHelper].user.name, newStr];
    if ([newStr isEqualToString:self.remark] || (!self.remark && newStr.length == 0)) {
        [self.navigationItem.rightBarButtonItem setTintColor:[UIColor jw_colorWithHex:0xb0b0b0]];
        self.navigationItem.rightBarButtonItem.enabled = NO;
    } else {
        [self.navigationItem.rightBarButtonItem setTintColor:[UIColor jw_colorWithHex:0x333333]];
        self.navigationItem.rightBarButtonItem.enabled = YES;
    }
    self.countLabel.text = [NSString stringWithFormat:@"%@/30", @([self convertToByte:newStr])];
    return YES;
}

一切看上去很完美,但是用系统输入法输入中文时,发现计算字节长度不对
比如输入dgzsg文本显示如图:
[图片上传失败...(image-b82596-1657457047127)]

发现诶?不对啊,明明几个英文加空格,怎么后面显示17/30,应该是9/30啊。
打断点看UITextField代理方法,打印newStr如下图:

[图片上传失败...(image-9b598a-1657457047127)]

也没问题啊,再看计算字节长度,发现转化完,p内容不对啊,怎么多了\x816\xa42这种不知道什么字符,如图:

[图片上传失败...(image-604e1b-1657457047127)]

这个应该是个特殊字符,但是不知道是什么,我们用编码查询,把newStr内容放上去,进行编码查询,如下图:

[图片上传失败...(image-b7f55c-1657457047127)]

我们直接输入个空格,进行编码查询

[图片上传失败...(image-4f553f-1657457047127)]

空格的unicode编码是0020,但是我们复制出来的空格unicode编码是2006,这就是问题所在了,系统键盘输入时,空格不是空格而是特殊字符。

找到问题,优化一下我们计算字节的方法,把这个特殊字符变成空格来计算我们长度。

// 计算字节的个数
- (NSUInteger)convertToByte:(NSString *)str {
    str = [str stringByReplacingOccurrencesOfString:@"\u2006" withString:@" "];
    NSStringEncoding encoding = CFStringConvertEncodingToNSStringEncoding(kCFStringEncodingGB_18030_2000);
    NSUInteger strLength = 0;
    char *p = (char *)[str cStringUsingEncoding:encoding];
    
    NSUInteger lengthOfBytes = [str lengthOfBytesUsingEncoding:encoding];
    for (int i = 0; i < lengthOfBytes; i++) {
        if (*p) {
            p++;
            strLength++;
        } else {
            p++;
        }
    }
    return strLength;
}

完成以上还有个细节需要处理,就是如果输入英文长度没达到限制,选择联想汉字长度超出时,按上面的字节数不会变化,如图

[图片上传失败...(image-72eab3-1657457047127)]

需要添加个通知:[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(textFieldChange) name:UITextFieldTextDidChangeNotification object:nil];

- (void)textFieldChange
{
    // 获取高亮内容的范围
    UITextRange *selectedRange = [self.textField markedTextRange];
    // 这行代码 可以认为是 获取高亮内容的长度
    NSInteger markedTextLength = [self.textField offsetFromPosition:selectedRange.start toPosition:selectedRange.end];
    // 没有高亮内容时,对已输入的文字进行操作
    if (markedTextLength == 0) {
        self.countLabel.text = [NSString stringWithFormat:@"%@/30", @([self convertToByte:self.textField.text])];
    }
}

本以为这样大功告成了,后来仔细测试发现自己太天真了。
发现两个bug。。。。

[图片上传失败...(image-6ec142-1657457047127)]

4个字母3个空格,应该是7/30为什么显示6/30呢?

[图片上传失败...(image-71b044-1657457047127)]

13个汉字,3个字母,2个空格,应该是31/30,而且超了限制。

发现每次输入键盘时,- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)stringstring都是单个字母,中间的空格是后面加上去的,所以这个代理方法不可靠了。修改通知方法:

- (void)textFieldChange
{
    NSInteger bitCount = [self convertToByte:self.textField.text];
    self.countLabel.text = [NSString stringWithFormat:@"%@/30", @(bitCount)];
    if (bitCount > 30) {
        [self.textField resignFirstResponder];
        [self.textField becomeFirstResponder];
    }
}

如果发现限制超了,直接resignFirstResponder把选中字母赋值上去,并且becomeFirstResponder,这样键盘就不会隐藏了。

最后附上demo:
Demo地址

你可能感兴趣的:(iOS 键盘联想特殊字符问题)