UITextView 限制输入字数

时间:2023-03-08 20:16:44
UITextView 限制输入字数

尊重原创  http://blog.csdn.net/fengsh998/article/details/45421107

对于限制UITextView输入的字符数。相信大家在网上见得最多的是实现UITextViewDelegate

  1. - (BOOL)textView:(UITextView *)textView shouldChangeTextInRange:(NSRange)range
  2. replacementText:(NSString *)text;//有输入时触但对于中文键盘出示的联想字选择时不会触发
  3. - (void)textViewDidChange:(UITextView *)textView;//当输入且上面的代码返回YES时触发。或当选择键盘上的联想字时触发。

第一个用于限制输入,第二个用于动态计算剩余字数。好吧,就来慢慢的给大家分析这两个代理共同协作来限制输入。

从最简单的开始。为了便于讲析,声明

#define MAX_LIMIT_NUMS     100 来限制最大输入只能100个字符

完整代码

  1. - (BOOL)textView:(UITextView *)textView shouldChangeTextInRange:(NSRange)range
  2. replacementText:(NSString *)text
  3. {
  4. UITextRange *selectedRange = [textView markedTextRange];
  5. //获取高亮部分
  6. UITextPosition *pos = [textView positionFromPosition:selectedRange.start offset:0];
  7. //获取高亮部分内容
  8. //NSString * selectedtext = [textView textInRange:selectedRange];
  9. //如果有高亮且当前字数开始位置小于最大限制时允许输入
  10. if (selectedRange && pos) {
  11. NSInteger startOffset = [textView offsetFromPosition:textView.beginningOfDocument toPosition:selectedRange.start];
  12. NSInteger endOffset = [textView offsetFromPosition:textView.beginningOfDocument toPosition:selectedRange.end];
  13. NSRange offsetRange = NSMakeRange(startOffset, endOffset - startOffset);
  14. if (offsetRange.location < MAX_LIMIT_NUMS) {
  15. return YES;
  16. }
  17. else
  18. {
  19. return NO;
  20. }
  21. }
  22. NSString *comcatstr = [textView.text stringByReplacingCharactersInRange:range withString:text];
  23. NSInteger caninputlen = MAX_LIMIT_NUMS - comcatstr.length;
  24. if (caninputlen >= 0)
  25. {
  26. return YES;
  27. }
  28. else
  29. {
  30. NSInteger len = text.length + caninputlen;
  31. //防止当text.length + caninputlen < 0时,使得rg.length为一个非法最大正数出错
  32. NSRange rg = {0,MAX(len,0)};
  33. if (rg.length > 0)
  34. {
  35. NSString *s = @"";
  36. //判断是否只普通的字符或asc码(对于中文和表情返回NO)
  37. BOOL asc = [text canBeConvertedToEncoding:NSASCIIStringEncoding];
  38. if (asc) {
  39. s = [text substringWithRange:rg];//因为是ascii码直接取就可以了不会错
  40. }
  41. else
  42. {
  43. __block NSInteger idx = 0;
  44. __block NSString  *trimString = @"";//截取出的字串
  45. //使用字符串遍历,这个方法能准确知道每个emoji是占一个unicode还是两个
  46. [text enumerateSubstringsInRange:NSMakeRange(0, [text length])
  47. options:NSStringEnumerationByComposedCharacterSequences
  48. usingBlock: ^(NSString* substring, NSRange substringRange, NSRange enclosingRange, BOOL* stop) {
  49. if (idx >= rg.length) {
  50. *stop = YES; //取出所需要就break,提高效率
  51. return ;
  52. }
  53. trimString = [trimString stringByAppendingString:substring];
  54. idx++;
  55. }];
  56. s = trimString;
  57. }
  58. //rang是指从当前光标处进行替换处理(注意如果执行此句后面返回的是YES会触发didchange事件)
  59. [textView setText:[textView.text stringByReplacingCharactersInRange:range withString:s]];
  60. //既然是超出部分截取了,哪一定是最大限制了。
  61. self.lbNums.text = [NSString stringWithFormat:@"%d/%ld",0,(long)MAX_LIMIT_NUMS];
  62. }
  63. return NO;
  64. }
  65. }
  66. - (void)textViewDidChange:(UITextView *)textView
  67. {
  68. UITextRange *selectedRange = [textView markedTextRange];
  69. //获取高亮部分
  70. UITextPosition *pos = [textView positionFromPosition:selectedRange.start offset:0];
  71. //如果在变化中是高亮部分在变,就不要计算字符了
  72. if (selectedRange && pos) {
  73. return;
  74. }
  75. NSString  *nsTextContent = textView.text;
  76. NSInteger existTextNum = nsTextContent.length;
  77. if (existTextNum > MAX_LIMIT_NUMS)
  78. {
  79. //截取到最大位置的字符(由于超出截部分在should时被处理了所在这里这了提高效率不再判断)
  80. NSString *s = [nsTextContent substringToIndex:MAX_LIMIT_NUMS];
  81. [textView setText:s];
  82. }
  83. //不让显示负数 口口日
  84. self.lbNums.text = [NSString stringWithFormat:@"%ld/%d",MAX(0,MAX_LIMIT_NUMS - existTextNum),MAX_LIMIT_NUMS];
  85. }

回顾一下,文章到此,共解决了哪些易遗留的BUG

1.中,英文字符输入时限制。

2.带emoji时截取显示半个或乱码字符处理。