转载自:http://blog.csdn.net/fengsh998/article/details/45421107
原文里面说得很详细,考虑了以下问题
1.中,英文字符输入时限制。
2.带emoji时截取显示半个或乱码字符处理。
废话不多说,最终完整代码
#import "ViewController.h"
#define SCREEN_WIDTH CGRectGetWidth([UIScreen mainScreen].bounds) // 屏宽
#define SCREEN_HEIGHT CGRectGetHeight([UIScreen mainScreen].bounds) // 屏高
#define MAX_LIMIT_NUMS 10 // 限制最大输入只能10个字符
@interface ViewController () <UITextViewDelegate>
@property (nonatomic, strong) UITextView *textView;
@property (nonatomic, strong) UILabel *countLabel;
@end
@implementation ViewController
- (void)viewDidLoad
{
[super viewDidLoad];
_textView = [[UITextView alloc] initWithFrame:CGRectMake(20, 80, SCREEN_WIDTH - 40, 100)];
_textView.delegate = self;
_textView.backgroundColor = [UIColor brownColor];
[self.view addSubview:_textView];
_countLabel = [[UILabel alloc] initWithFrame:CGRectMake(SCREEN_WIDTH - 60, CGRectGetMaxY(_textView.frame), 60, 20)];
_countLabel.text = [NSString stringWithFormat:@"%d/%ld", 0, (long)MAX_LIMIT_NUMS];
[self.view addSubview:_countLabel];
}
实现UITextViewDelegate的两个协议方法
// 用于动态计算剩余字数
- (void)textViewDidChange:(UITextView *)textView {
UITextRange *selectedRange = [textView markedTextRange];
// 获取高亮部分
UITextPosition *pos = [textView positionFromPosition:selectedRange.start offset:0];
// 如果在变化中是高亮部分在变,就不计算字符
if (selectedRange && pos) {
return;
}
NSString *nsTextContent = textView.text;
NSInteger existTextNum = nsTextContent.length;
if (existTextNum > MAX_LIMIT_NUMS) {
// 截取到最大位置的字符(由于超出截部分在should时被处理了所在这里这了提高效率不再判断)
NSString *s = [nsTextContent substringToIndex:MAX_LIMIT_NUMS];
textView.text = s;
existTextNum = textView.text.length;
}
// 不让显示负数
_countLabel.text = [NSString stringWithFormat:@"%ld/%d", existTextNum, MAX_LIMIT_NUMS];
}
// 用于限制输入
- (BOOL)textView:(UITextView *)textView shouldChangeTextInRange:(NSRange)range replacementText:(NSString *)text
{
UITextRange *selectedRange = [textView markedTextRange];
// 获取高亮部分
UITextPosition *pos = [textView positionFromPosition:selectedRange.start offset:0];
// 获取高亮部分内容
//NSString * selectedtext = [textView textInRange:selectedRange];
// 如果有高亮且当前字数开始位置小于最大限制时允许输入
if (selectedRange && pos) {
NSInteger startOffset = [textView offsetFromPosition:textView.beginningOfDocument toPosition:selectedRange.start];
NSInteger endOffset = [textView offsetFromPosition:textView.beginningOfDocument toPosition:selectedRange.end];
NSRange offsetRange = NSMakeRange(startOffset, endOffset - startOffset);
if (offsetRange.location < MAX_LIMIT_NUMS) {
return YES;
} else {
return NO;
}
} else {
NSString *comcatstr = [textView.text stringByReplacingCharactersInRange:range withString:text];
NSInteger caninputlen = MAX_LIMIT_NUMS - comcatstr.length;
if (caninputlen >= 0)
{
return YES;
} else {
NSInteger len = text.length + caninputlen;
// 防止当text.length + caninputlen < 0时,使得rg.length为一个非法最大正数出错
NSRange rg = {0,MAX(len,0)};
if (rg.length > 0) {
NSString *s = @"";
// 判断是否只普通的字符或asc码(对于中文和表情返回NO)
BOOL asc = [text canBeConvertedToEncoding:NSASCIIStringEncoding];
if (asc) {
s = [text substringWithRange:rg]; // 因为是ascii码直接取就可以了不会错
} else {
__block NSInteger idx = 0;
__block NSString *trimString = @"";//截取出的字串
// 使用字符串遍历,这个方法能准确知道每个emoji是占一个unicode还是两个
[text enumerateSubstringsInRange:NSMakeRange(0, [text length])
options:NSStringEnumerationByComposedCharacterSequences
usingBlock: ^(NSString* substring, NSRange substringRange, NSRange enclosingRange, BOOL* stop) {
NSInteger steplen = substring.length;
if (idx >= rg.length) {
*stop = YES; // 取出所需要就break,提高效率
return ;
}
trimString = [trimString stringByAppendingString:substring];
idx = idx + steplen; // 这里变化了,使用了字串占的长度来作为步长
}];
s = trimString;
}
// rang是指从当前光标处进行替换处理(注意如果执行此句后面返回的是YES会触发didchange事件)
textView.text = [textView.text stringByReplacingCharactersInRange:range withString:s];
// 既然是超出部分截取了,哪一定是最大限制了。
_countLabel.text = [NSString stringWithFormat:@"%d/%ld",0,(long)MAX_LIMIT_NUMS];
}
return NO;
}
}
}