UITextView 限制输入文本长度

转载自: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;
        }
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
UITextView是UIKit框架中的一个控件,可以用来显示和编辑长文本。而富文本则是指带有丰富样式的文本,可以设置文字的字体、颜色、大小、间距、行高等属性。 要在UITextView中实现富文本,首先需要创建一个NSAttributedString对象,并通过NSMutableAttributedString来设置文字的样式。NSAttributedString是不可变的,而NSMutableAttributedString可以修改和添加样式。 创建NSMutableAttributedString对象后,可以使用其方法来设置文字的样式,比如设置字体可以使用NSFontAttributeName属性,设置颜色可以使用NSForegroundColorAttributeName属性,设置字号可以使用NSFontAttributeName属性,设置段落样式可以使用NSParagraphStyleAttributeName属性等等。 设置完成后,就可以将NSMutableAttributedString对象赋值给UITextView的attributedText属性,以实现富文本的显示。 例如,我们想将某个UITextView的文字样式设置为红色、字号为20、字体为粗体,可以按如下方式设置: ``` NSMutableAttributedString *attributedString = [[NSMutableAttributedString alloc] initWithString:@"这是富文本"]; [attributedString addAttribute:NSForegroundColorAttributeName value:[UIColor redColor] range:NSMakeRange(0, attributedString.length)]; [attributedString addAttribute:NSFontAttributeName value:[UIFont boldSystemFontOfSize:20] range:NSMakeRange(0, attributedString.length)]; textView.attributedText = attributedString; ``` 通过上述代码,就可以在UITextView中显示带有红色、字号为20、字体为粗体的文字。 除了以上示例外,UITextView还支持更多的富文本样式设置,根据具体需求,可以设置更多的属性来实现更丰富的文本效果。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值