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
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
//
// UITextView+LengthLimit.m
// Gengmei
//
// Created by wangyang on 2017/3/9.
// Copyright © 2017年 更美互动信息科技有限公司. All rights reserved.
//
#import "UITextView+LengthLimit.h"
#import <objc/runtime.h>
@implementation UITextView (LimitBase)
- (NSInteger)maxTextLength {
NSNumber *length = objc_getAssociatedObject(self, @selector(maxTextLength));
return length.integerValue;
}
- (void)setMaxTextLength:(NSInteger)maxTextLength {
objc_setAssociatedObject(self, @selector(maxTextLength), @(maxTextLength), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (NSInteger)textLength {
NSNumber *length = objc_getAssociatedObject(self, @selector(textLength));
return length.integerValue;
}
- (void)setTextLength:(NSInteger)textLength {
objc_setAssociatedObject(self, @selector(textLength), @(textLength), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
@end
@implementation UITextView (SimpleLimit)
- (void)s_textViewDidChange {
NSString *toBeString = self.text;
UITextRange *selectedRange = [self markedTextRange];
UITextPosition *position = [self positionFromPosition:selectedRange.start offset:0];
NSInteger length = self.textLength;
// 没有高亮选择的字,则对已输入的文字进行字数统计和限制
if (!position){
if (toBeString.length > self.maxTextLength){
NSRange rangeIndex = [toBeString rangeOfComposedCharacterSequenceAtIndex:self.maxTextLength];
if (rangeIndex.length == 1){
self.text = [toBeString substringToIndex:self.maxTextLength];
} else{
NSRange rangeRange = [toBeString rangeOfComposedCharacterSequencesForRange:NSMakeRange(0, self.maxTextLength)];
self.text = [toBeString substringWithRange:rangeRange];
}
length = self.text.length;
} else {
NSString *selectedText = [self textInRange:selectedRange];
length = self.text.length - selectedText.length;
}
}
if (length > self.maxTextLength) {
self.textLength = self.maxTextLength;
} else {
self.textLength = length;
}
}
@end
@implementation UITextView (ComplexLimit)
- (BOOL)c_shouldChangeTextInRange:(NSRange)range replacementText:(NSString *)text
{
if (text.length==0) { //Delete any cases
if(range.length > 1){
//Delete whole word
}
else if(range.length == 1){
//Delete single letter
}
else if(range.length == 0){
//Tap delete key when textField empty
}
return YES;
}
UITextRange *selectedRange = [self markedTextRange];
//获取高亮部分
UITextPosition *pos = [self positionFromPosition:selectedRange.start offset:0];
//获取高亮部分内容
// NSString * selectedtext = [self textInRange:selectedRange];
//如果有高亮且当前字数开始位置小于最大限制时允许输入
if (selectedRange && pos) {
NSInteger startOffset = [self offsetFromPosition:self.beginningOfDocument toPosition:selectedRange.start];
NSInteger endOffset = [self offsetFromPosition:self.beginningOfDocument toPosition:selectedRange.end];
NSRange offsetRange = NSMakeRange(startOffset, endOffset - startOffset);
if (offsetRange.location < self.maxTextLength) {
return YES;
}
else
{
return NO;
}
}
NSString *comcatstr = [self.text stringByReplacingCharactersInRange:range withString:text];
NSInteger canInputLen = self.maxTextLength - 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事件)
[self setText:[self.text stringByReplacingCharactersInRange:range withString:s]];
//既然是超出部分截取了,那一定是最大限制了。
self.textLength = self.maxTextLength;
}
return NO;
}
}
- (void)c_textViewDidChange
{
UITextRange *selectedRange = [self markedTextRange];
//获取高亮部分
UITextPosition *pos = [self positionFromPosition:selectedRange.start offset:0];
//如果在变化中是高亮部分在变,就不要计算字符了
if (selectedRange && pos) {
return;
}
NSString *nsTextContent = self.text;
NSInteger existTextNum = nsTextContent.length;
if (existTextNum > self.maxTextLength)
{
//截取到最大位置的字符(由于超出截部分在should时被处理了所在这里这了提高效率不再判断)
NSString *s = [nsTextContent substringToIndex:self.maxTextLength];
[self setText:s];
self.textLength = self.maxTextLength;
} else {
self.textLength = existTextNum;
}
}
@end