前言
这是在我项目开发中遇到的问题,主要是给创建的作品保存名字,这样我们就会涉及到名称字数和特殊字符的限制,目前涉及到用户输入的大部分解决方案就是UITextField和UITextView,两者的原理差不多,那么下面就针对UITextField进行简单的分享.
补充:目前为止XCode版本更新后(目前为8.1)对开发者账户进行了相关的限制,至于何种限制,目前影响最大的就是从网上下载下来的项目不能用了,原因是证书等问题,所以给大家一个建议,小程序可以用模拟器的用模拟器,用到真机的尽可能的选择将你开发组的账号,这个一般公司都会配备.这里简单说一下,详细的可以参考我的另外一边文章:iOS坑--XCode8 的一些问题,希望对大家有所帮助.
正文
1.准备工作
1).创建属性
由于项目中的UITextField为StoryBoard控件,创建省略
/// 记录显示剩余个数
@property (weak, nonatomic) IBOutlet UILabel *totalCharacterLabel;
/// 输入UITextField
@property (weak, nonatomic) IBOutlet UITextField *nameField;
2).设置最大限制数
static NSInteger CharacterCount = 8;
3).设置代理
self.nameField.delegate = self;
4).遵循代理
@interface ViewController () <UITextFieldDelegate>
2.原理
首先我们需要区分的是UITextField的输入分为两个阶段,即为输入阶段和确定输入阶段,输入阶段这里主要针对的是中文的输入,这里不需要做严格的判断要求,再就是确定输入阶段,也就是UITextField的输出显示阶段,这里需要排除空格,emoji等特殊字符和字数的限制(截取有效字符).
3.观察者
1).添加观察者
[self.nameField addTarget:self action:@selector(textFieldChanged:)forControlEvents:UIControlEventEditingChanged];
2).执行观察者对应的方法
在方法中,特地针对目前iOS支持的第三方输入法做出的解决方案
- (void)textFieldChanged:(UITextField *)textField {
NSString *toBeString = textField.text;
if (![self isInputRuleAndBlank:toBeString]) {
textField.text = [self disable_emoji:toBeString];
return;
}
NSString *lang = [[textField textInputMode] primaryLanguage]; // 获取当前键盘输入模式
//简体中文输入,第三方输入法(搜狗)所有模式下都会显示“zh-Hans”
if([lang isEqualToString:@"zh-Hans"]) {
UITextRange *selectedRange = [textField markedTextRange];
//获取高亮部分
UITextPosition *position = [textField positionFromPosition:selectedRange.start offset:0];
//没有高亮选择的字,则对已输入的文字进行字数统计和限制
if(!position) {
NSString *getStr = [self getSubString:toBeString];
if(getStr && getStr.length > 0) {
textField.text = getStr;
}
}
} else{
NSString *getStr = [self getSubString:toBeString];
if(getStr && getStr.length > 0) {
textField.text= getStr;
}
}
}
4.执行代理
1). 在UITextField的代理中,主要是对确定输入阶段的判断,确定没有空格和空
- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string {
if ([self isInputRuleNotBlank:string] || [string isEqualToString:@""]) {//当输入符合规则和退格键时允许改变输入框
return YES;
} else {
NSLog(@"超出字数限制");
return NO;
}
}
2). return收回键盘
- (BOOL)textFieldShouldReturn:(UITextField *)textField {
[self.view endEditing:YES];
return YES;
}
5.输入内容判断(正则判断)
1).字母、数字、中文正则判断(不包括空格)
- (BOOL)isInputRuleNotBlank:(NSString *)str {
NSString *pattern = @"^[a-zA-Z\u4E00-\u9FA5\\d]*$";
NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", pattern];
BOOL isMatch = [pred evaluateWithObject:str];
return isMatch;
}
2).字母、数字、中文正则判断(包括空格)(在系统输入法中文输入时会出现拼音之间有空格,需要忽略,当按return键时会自动用字母替换,按空格输入响应汉字)
注意: 因为考虑到输入习惯,许多人习惯使用九宫格,这里在正常选择全键盘输入错误的时候,进行九宫格判断,九宫格对应的是下面➋➌➍➎➏➐➑➒的字符.目前项目已经在github上进行更新.
- (BOOL)isInputRuleNotBlank:(NSString *)str {
NSString *pattern = @"^[a-zA-Z\u4E00-\u9FA5\\d]*$";
NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", pattern];
BOOL isMatch = [pred evaluateWithObject:str];
// 这里是后期补充的内容:九宫格判断
if (!isMatch) {
NSString *other = @"➋➌➍➎➏➐➑➒";
unsigned long len=str.length;
for(int i=0;i<len;i++)
{
unichar a=[str characterAtIndex:i];
if(!((isalpha(a))
||(isalnum(a))
||((a=='_') || (a == '-'))
||((a >= 0x4e00 && a <= 0x9fa6))
||([other rangeOfString:str].location != NSNotFound)
))
return NO;
}
return YES;
}
return isMatch;
}
3).过滤字符串中的emoji
- (NSString *)disable_emoji:(NSString *)text {
NSRegularExpression *regex = [NSRegularExpression regularExpressionWithPattern:@"[^\\u0020-\\u007E\\u00A0-\\u00BE\\u2E80-\\uA4CF\\uF900-\\uFAFF\\uFE30-\\uFE4F\\uFF00-\\uFFEF\\u0080-\\u009F\\u2000-\\u201f\r\n]"options:NSRegularExpressionCaseInsensitive error:nil];
NSString *modifiedString = [regex stringByReplacingMatchesInString:text
options:0
range:NSMakeRange(0, [text length])
withTemplate:@""];
return modifiedString;
}
6.截取字符串(重点)
有趣的是,根据不同的需求做出了两种不同的方案,就是中文按照原理是占用两个字节来算,但是有很多需求只是简单地计算个数,即为表面上看到的个数,所以从用户角度来讲,个数更符合用户审美.但是从一个程序员角度而言,两个限制一个中文,没错,程序员,就是这么矫情,反正,下面做出了两种不同的解决方案,根据你的需求而定吧
-(NSString *)getSubString:(NSString*)string
{
/// 第一种:两个字节一个中文
// NSStringEncoding encoding = CFStringConvertEncodingToNSStringEncoding(kCFStringEncodingGB_18030_2000);
// NSData* data = [string dataUsingEncoding:encoding];
// NSInteger length = [data length];
// if (length > CharacterCount) {
// NSData *data1 = [data subdataWithRange:NSMakeRange(0, CharacterCount)];
// NSString *content = [[NSString alloc] initWithData:data1 encoding:encoding];//注意:当截取CharacterCount长度字符时把中文字符截断返回的content会是nil
// if (!content || content.length == 0) {
// data1 = [data subdataWithRange:NSMakeRange(0, CharacterCount - 1)];
// content = [[NSString alloc] initWithData:data1 encoding:encoding];
// }
// return content;
// }
// return nil;
/// 第二种:按照个数进行判断
if (string.length > CharacterCount) {
NSLog(@"超出字数上限");
_totalCharacterLabel.text = @"0";
return [string substringToIndex:CharacterCount];
}else {
_totalCharacterLabel.text = [NSString stringWithFormat:@"%ld",(long)(CharacterCount - string.length)];
}
return nil;
}
其他
点击view的其他区域收回键盘
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
[self.view endEditing:YES];
}
写在最后
- 项目中遇到的问题,给予的解决方案,可能不是做完美的,但是根据需求已经达到要求,所以,如有不做指出,欢迎大家相互交流
- 此次解决方案中,用到的正则表达式没有过多性的进行表述,因为本人对这方面还未曾涉猎,今后会做以补充
- 按照惯例,奉上Demo地址:
zhangfurun的Github--UITextFieldDemo-限制字数