最近项目开发中用到了多行文本框限制文字输入个数限制,之前的话在textViewDidChange方法中截取超出字数就可以。测试人员发现在拼音转汉字的过程中,YYTextView不会二次确认textViewDidChange方法,输入字数只能最后确认结束弹框的时候计算,影响体验效果,后转用自己封装的继承UITextView替换。另外拼音转汉字的情况下,在输入文本中间一直输入,依然会出现输入字数计算不准确的问题。下面列出解决问题实现的代码。

  

#pragma mark- UITextViewDelegate

-(void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event

{

[self.activityTextView resignFirstResponder];

}

-(void)updateSpeechButtonStatus{

if (self.activityTextView.text.length >= MAX_NUM) {

self.btnSpeech.enabled = NO;

}else{

self.btnSpeech.enabled = YES;

}

}

//动态显示可输入文本字数

-(void)changeTextNumber:(UITextView *)textView{

if (textView.text.length <= MAX_NUM) {

self.numLabel.text = [NSString stringWithFormat:@"%lu/%lu",(unsigned long)textView.text.length,(MAX_NUM - textView.text.length)];

}else{

self.numLabel.text = [NSString stringWithFormat:@"%@/%@",[NSString stringWithFormat:@"%d",MAX_NUM],@"0"];

}

}

-(void)textViewDidChange:(UITextView *)textView{

UITextRange *selectedRange = [textView markedTextRange];

//获取高亮部分

UITextPosition *pos = [textView positionFromPosition:selectedRange.start offset:0];

//如果是拼音转汉语状态不统计字数,确认输入后统计

if (selectedRange && pos) {

return;

}

[self changeTextNumber:textView];

[self updateSpeechButtonStatus];

}

-(void)textViewDidEndEditing:(UITextView *)textView{

textView.text = [EaseEmoji stringReplaceEmoji:textView.text];

[self changeTextNumber:textView];

[self updateSpeechButtonStatus];

}

- (BOOL)textView:(UITextView *)textView shouldChangeTextInRange:(NSRange)range replacementText:(NSString *)text

{

BOOL isLimit = [CCCommonAPI textViewTextCountLimit:textView range:range text:text maxCount:[NSString stringWithFormat:@"%d",MAX_NUM] isAllowEmoji:YES completeBlock:^{

//执行这里block的时候不执行textViewDidChange,block里面的内容就是textViewDidChange里面的操作

[self changeTextNumber:textView];

[self updateSpeechButtonStatus];

}];

return isLimit;

}

//textview字数限制处理

+ (BOOL)textViewTextCountLimit:(UITextView *)textView range:(NSRange)range text:(NSString *)text maxCount:(NSString *)maxCountStr isAllowEmoji:(BOOL)isAllowEmoji completeBlock:(nonnull CommonTextLimitCompletion)completeBlock{

if (!isAllowEmoji) {

//不允许输入表情的时候返回NO

if ([[[UITextInputMode currentInputMode] primaryLanguage] isEqualToString:@"emoji"]) {

return NO;

}

}

//对于退格删除键开放限制

if (text.length == 0) {return YES;}

NSString *comcatstr = [textView.text stringByReplacingCharactersInRange:range withString:text];

NSInteger caninputlen = [maxCountStr intValue] - 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) {

if (idx >= rg.length) {

*stop = YES; //取出所需要就break,提高效率

return ;

}

trimString = [trimString stringByAppendingString:substring];

idx++;

}];

s = trimString;

}

//rang是指从当前光标处进行替换处理(注意如果执行此句后面返回的是YES会触发didchange事件)

[textView setText:[textView.text stringByReplacingCharactersInRange:range withString:s]];

completeBlock();

//手动截取更改,不再调用didchange

return NO;

}else{

UITextRange *selectedRange = [textView markedTextRange];

//获取高亮部分

UITextPosition *pos = [textView positionFromPosition:selectedRange.start offset:0];

if (selectedRange && pos) {

//如果拼音转汉语状态继续输入

return YES;

}else{

return NO;

}

}

}

}

iOS开发中UITextView使用小结的更多相关文章

  1. IOS开发中设置控件内容对齐方式时容易混淆的几个属性

    IOS开发中四个容易混淆的属性: 1. textAligment : 文字的水平方向的对齐方式 1> 取值 NSTextAlignmentLeft      = 0,    // 左对齐 NST ...

  2. iOS开发中你是否遇到这些经验问题

    前言 小伙伴们在开发中难免会遇到问题, 你是如何解决问题的?不妨也分享给大家!如果此文章其中的任何一条问题对大家有帮助,那么它的存在是有意义的! 反正不管怎样遇到问题就要去解决问题, 在解决问题的同时 ...

  3. iOS开发中遇到的一些问题及解决方案【转载】

    iOS开发中遇到的一些问题及解决方案[转载] 2015-12-29 [385][scrollView不接受点击事件,是因为事件传递失败] // //  MyScrollView.m //  Creat ...

  4. ios开发中关闭textview控件的虚拟键盘

    在ios开发中,textfield控件在点击的时候出现虚拟键盘,关掉虚拟键盘可以通过虚拟键盘中的done button和点击view中的任意地方来关闭虚拟键盘. 1.第一种方法是textfield控件 ...

  5. 再续iOS开发中的这些权限

    前言 上篇文章iOS开发中的这些权限,你搞懂了吗?介绍了一些常用权限的获取和请求方法,知道这些方法的使用基本上可以搞定大部分应用的权限访问的需求.但是,这些方法并不全面,不能涵盖住所有权限访问的方法. ...

  6. iOS开发中权限再度梳理

    前言 上篇文章iOS开发中的这些权限,你搞懂了吗?介绍了一些常用权限的获取和请求方法,知道这些方法的使用基本上可以搞定大部分应用的权限访问的需求.但是,这些方法并不全面,不能涵盖住所有权限访问的方法. ...

  7. 总结iOS开发中的断点续传那些事儿

    前言 断点续传概述 断点续传就是从文件赏赐中断的地方重新开始下载或者上传数据,而不是从头文件开始.当下载大文件的时候,如果没有实现断点续传功能,那么每次出现异常或者用户主动的暂停,都会从头下载,这样很 ...

  8. iOS开发中静态库之".framework静态库"的制作及使用篇

    iOS开发中静态库之".framework静态库"的制作及使用篇 .framework静态库支持OC和swift .a静态库如何制作可参照上一篇: iOS开发中静态库之" ...

  9. iOS开发中静态库制作 之.a静态库制作及使用篇

    iOS开发中静态库之".a静态库"的制作及使用篇 一.库的简介 1.什么是库? 库是程序代码的集合,是共享程序代码的一种方式 2.库的类型? 根据源代码的公开情况,库可以分为2种类 ...

  10. ios开发中的小技巧

    在这里总结一些iOS开发中的小技巧,能大大方便我们的开发,持续更新. UITableView的Group样式下顶部空白处理 //分组列表头部空白处理 UIView *view = [[UIViewal ...

随机推荐

  1. 9、IDEA集成Github

    9.1.登录Github账号 9.1.1.打开IDEA的Settings界面 如上图所示,打开IDEA的 Settings(设置)界面. 9.1.2.使用账号密码登录(方式一) 如上图所示,在&quo ...

  2. 【Java】逻辑错误BUG

    开局一张图来解释就够了 查询 COUNT() 结果数,有且仅有一条记录 好死不死判断查询的结果数量等等于0, 这不永远都是取TRUE返回 花了一个下午的时间就为了解决这个BUG

  3. 华为显卡已经支持pytorch计算框架

    相关链接: https://support.huawei.com/enterprise/zh/doc/EDOC1100079287/a21c08de https://www.zhihu.com/que ...

  4. python语言绘图:绘制一组二项分布图

    代码源自: https://github.com/PacktPublishing/Bayesian-Analysis-with-Python ============================= ...

  5. 【安装】SQL SERVER 彻底卸载清除

    -----2024年8月6日09:40:13 -----bayaim,  以下内容纯属百度网络搜到,如有侵权请联系及时删除 SQL SERVER 如果卸载不干净,就会导致下一次安装失败,下面是卸载的步 ...

  6. blender-十大基本操作

  7. Java静态相关问题

    问题1: public class Demo01_StaticTest { private static Demo01_StaticTest st = new Demo01_StaticTest(); ...

  8. Django框架创建运行最小程序过程记录

    基于 python语言 Django web框架下 用pycharm创建,修改,运行 最简单程序.旨在过程 ========================================== 步骤一 ...

  9. 关于Vue + element plus包装Component理解

    关于Vue + element plus包装Component理解 一.关于编写思路 我以设计el-select选择框进行举例说明 父组件与Component传递params与Function使用Pr ...

  10. IE、Chrome、Firefox修改http header信息

    在测试系统交互时,可能会碰到需要修改header信息的要求,下面介绍下如何在IE.Chrome.Firefox修改http header信息. 1.IE(fiddler) >在IE下修改head ...