在第三方键盘加入之后,对于字数限制的处理不再像之前那么简单了

纯数字、字符输入(不包括粘贴)这样的字数限制还是相对比较简单的,你可以用两种方法进行处理

第一种是textfield的delegate实现:

1
2
3
4
5
6
7
- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string {
if (range.length + range.location > textField.text.length) {
return NO;
}
NSUInteger newLength = textField.text.length + string.length - range.length;
return newLength<=kMaxCharacterCount;
}

第二种是注册一个通知,在textfield编辑时做处理:

首先你在viewDidLoad中注册通知

1
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(textFiledEditChanged:) name:UITextFieldTextDidChangeNotification object:self.shopName];

再实现通知里面的方法,在超过最大值时,取最大的字数

1
2
3
if (self.shopName.text.length > kMaxCharacterCount) {
self.shopName.text = [self.shopName.text substringToIndex:kMaxCharacterCount];
}

但是,在中文的限制上面情况就复杂了,当时在调试的时候,因为使用的是第三方键盘,所以当时没有发现问题,但是在使用系统键盘的时候,一下子就蛋疼了….

下面我开始分析一下,两者的区别:

1、第三方键盘在输入字符时,一般是不会将字符直接输入到textfield中,而是将字符显示在它自己的view上方,但是系统键盘会直接输入到textfield中,而且它会占2个字符长度,比如你输入”abcd”,在textfield中显示的是”a b c d”,并且”a b c d”是处在高亮中的,并不算是真正输入到textfield中,所以我们不应把高亮的字符计算在内,我们应该计算真正输入的字符

2、如果我们使用的是delegate做处理的时候,系统中文输入的时候会有联想,但是联想的那个字并不会调用delegate,比如你输入一个”你”,在系统的联想里面可能会出现”的”,”们”这样的联想,但是你选择”的”的时候,delegate并不会调用,(尼玛….),我猜想联想输入应该不算做keyboard所触发的事件,所以他并不会触发delegate,但是如果你注册的是通知,他倒是会调用,(还好有救)

另外提醒一下,有时候在自测输入的时候,要考虑全面,比如粘贴这也是一种输入,当时没考虑,我也是跪了

好了,分析了主要的区别下面我们就来看看具体怎么实现吧~

在实现注册通知方法里面:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
#pragma mark - UITextViewDelegate
- (void)textViewDidChange:(UITextView *)textView {
if (textView.text.length == 0) {
self.recommendTips.hidden = NO;
}else{
self.recommendTips.hidden = YES;
}

NSString *toBeString = textView.text;
NSString *lang = self.textInputMode.primaryLanguage; // 键盘输入模式
if ([lang isEqualToString:@"zh-Hans"]) { // 简体中文输入,包括简体拼音,健体五笔,简体手写
UITextRange *selectedRange = [textView markedTextRange];
//获取高亮部分
UITextPosition *position = [textView positionFromPosition:selectedRange.start offset:0];
// 没有高亮选择的字,则对已输入的文字进行字数统计和限制
if (!position || !selectedRange) {
if (toBeString.length > 200) {
textView.text = [toBeString substringToIndex:200];
}
}
// 有高亮选择的字符串,则暂不对文字进行统计和限制
else{

}
}
// 中文输入法以外的直接对其统计限制即可,不考虑其他语种情况
else{
if (toBeString.length > 200) {
textView.text = [toBeString substringToIndex:200];
}
}
}

首先,我们根据键盘的输入模式进行区分,英文的比较简单就和上面一样,直接取最大的字符数就好了,在中文输入的时候,我们用markedTextRange方法获取到当前的光标位置,再用textField positionFromPosition:selectedRange.start offset:0获取到高亮部分,然后判断是否有高亮,这个时候系统会调用两次通知方法,第一次是将高亮的字符输入,第二次是将高亮的字符转换成中文输入(这个时候就没有高亮了,然后再取最大的字符数),但是在iOS7的设备上测试时发现,position都不会为nil,在iOS8以上都正常,但是获取到光标的range,却是正常的

1
2
3
4
5
6
7
NS_CLASS_AVAILABLE_IOS(3_2) @interface UITextRange : NSObject

@property (nonatomic, readonly, getter=isEmpty) BOOL empty; // Whether the range is zero-length.
@property (nonatomic, readonly) UITextPosition *start;
@property (nonatomic, readonly) UITextPosition *end;

@end

我们可以看到系统的UITextRange,有两个变量,一个是start,一个是end,这正是对于的高亮区域!

所以既然position不能使用,那我们干脆就使用range,通过判断range的存在,来对文字进行限制处理。(粘贴也适用)

结果也是棒棒的!