话不多说先上图
[图片上传失败...(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 *)string
的string
都是单个字母,中间的空格是后面加上去的,所以这个代理方法不可靠了。修改通知方法:
- (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地址