iOS聊天表情的实现

//内容处理的方法,继承于UITextView的类(即self为UITextView实例对象), 返回一个处理好的UITextView(包括链接,电话,表情排列等)

//表情处理方法一

NSDictionary *attributes =@{NSFontAttributeName:[UIFont systemFontOfSize:textFont]};//textFont文字大小

NSMutableAttributedString *mutableAttribute = [[NSMutableAttributedString alloc]initWithString:contentText attributes:attributes];//contentText聊天内容

//表情(_imagesRegexString为表情正则)

if(_imagesRegexString) {

NSRegularExpression *emotionExpression = [NSRegularExpression regularExpressionWithPattern:_imagesRegexString options:NSRegularExpressionCaseInsensitive error:nil];

NSArray*matchs = [emotionExpression matchesInString:contentText options:0range:NSMakeRange(0, contentText.length)];

intrangeDelete =0;//删除一个表情字符后range的location要前移

for(NSTextCheckingResult *result in matchs) {

NSRange range = [result range];

NSString*currentEmojiString = [contentText substringWithRange:range];//表情显示字符

NSString*emotionImageName = [self loadEmotionImage:currentEmojiString];//表情图片名称

UIImage*image = [UIImage imageNamed:emotionImageName];

if(emotionImageName.length!=0&& image) {

NSTextAttachment*attachment = [[NSTextAttachment alloc]init];

attachment.image= [UIImage imageNamed:emotionImageName];

attachment.bounds=CGRectMake(0, -5, emotionSize.width, emotionSize.height);

[mutableAttribute deleteCharactersInRange:NSMakeRange(range.location- rangeDelete, range.length)];

NSAttributedString*attributeString = [NSAttributedString attributedStringWithAttachment:attachment];

[mutableAttribute insertAttributedString:attributeStringatIndex:range.location- rangeDelete];

rangeDelete += range.length-1;//前移删除表情字符长度的累加(插入的表情图片占1个长度)

//表情处理方法二

NSMutableString *emojiString = [NSMutableString stringWithString:contentText];

NSMutableAttributedString *mutableAttribute = [[NSMutableAttributedString alloc] init];

NSMutableArray *attributes = [NSMutableArray array];

//根据正则一次获取符合的字符结果

NSTextCheckingResult *result = [expression firstMatchInString:emojiString options:NSMatchingReportProgress range:NSMakeRange(0, emojiString.length)];

while (result) {

NSRange range = [result rangeAtIndex:0];

NSString *prefixNotEmojiString = [emojiString substringToIndex:range.location];

NSString *currentEmojiString = [emojiString substringWithRange:range];

if (prefixNotEmojiString.length) {

NSAttributedString *attributeString = [[NSAttributedString alloc] initWithString:prefixNotEmojiString attributes:@{NSFontAttributeName:[UIFont systemFontOfSize:textFont]}];

[attributes addObject:attributeString];

}

if (currentEmojiString.length) {

NSString *emotionImageName = [self loadEmotionImage:currentEmojiString];

UIImage *image = [UIImage imageNamed:emotionImageName];

//表情名称字符不存在或表情图片不存在,直接显示字符而不是表情

if (emotionImageName.length == 0 || !image) {

NSAttributedString *attributeString = [[NSAttributedString alloc] initWithString:currentEmojiString attributes:@{NSFontAttributeName:[UIFont systemFontOfSize:textFont]}];

[attributes addObject:attributeString];

}else{

NSTextAttachment *attachment = [[NSTextAttachment alloc] init];

attachment.image = [UIImage imageNamed:emotionImageName];

attachment.bounds = CGRectMake(0, -5, emotionSize.width, emotionSize.height);

NSAttributedString *attributeString = [NSAttributedString attributedStringWithAttachment:attachment];

[attributes addObject:attributeString];

}

}

//删除已经处理的,更新正则结果

[emojiString deleteCharactersInRange:NSMakeRange(0, range.location+range.length)];

result = [expression firstMatchInString:emojiString options:NSMatchingReportProgress range:NSMakeRange(0, emojiString.length)];

}

//最后添加上后面的非表情字符

NSAttributedString *attributeString = [[NSAttributedString alloc]initWithString:emojiString];

[attributes addObject:attributeString];

[attributes enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {

[mutableAttribute appendAttributedString:obj];

}];



//@功能(_singleRegexString @正则)

if(_singleRegexString) {

NSRegularExpression*singleExpression = [NSRegularExpression regularExpressionWithPattern:_singleRegexString options:NSRegularExpressionCaseInsensitive error:nil];

NSArray*singlematchs = [singleExpression matchesInString:[mutableAttribute string]options:0range:NSMakeRange(0, mutableAttribute.length)];

if(singlematchs.count>0) {

NSRange range = [singlematchs[0] range];

if(range.location==0) {//@功能只能在第一个字符

NSString*singleString = [[mutableAttribute string] substringWithRange:range];

if(singleString.length<=24) {//@的人名字限定在24个字符

[mutableAttribute addAttribute:NSForegroundColorAttributeName value:_singleColor range:range];

[mutableAttribute addAttribute:NSUnderlineStyleAttributeName value:@(NSUnderlineStyleNone) range:range];

}

}

}

}

//行间距(_lineSpace默认值为0)

if(_lineSpace) {

NSMutableParagraphStyle* paragraphStyle = [[NSMutableParagraphStyle alloc]init];

[paragraphStyle setLineSpacing:_lineSpace];

[mutableAttribute addAttribute:NSParagraphStyleAttributeName value:paragraphStyle range:NSMakeRange(0, [mutableAttribute length])];

}

//设置连接textView自带检索连接,电话,地址等功能,无需手动设置

self.delegate=self;

self.dataDetectorTypes=UIDataDetectorTypeLink|UIDataDetectorTypePhoneNumber;

self.editable=NO;//非编辑状态下才可以点击Url

self.scrollEnabled=NO;

self.attributedText= mutableAttribute;

CGSizesize = [self sizeThatFits:CGSizeMake(contentWidth,CGFLOAT_MAX)];

self.size= size;


//UITextView代理,匹配电话/链接等

- (BOOL)textView:(UITextView*)textView shouldInteractWithURL:(NSURL*)URL inRange:(NSRange)characterRange {

NSLog(@"url :%@",URL);

//点击电话号码

if([[URLscheme]isEqualToString:@"tel"]) {

NSString*telephone = [URLresourceSpecifier];

NSLog(@"电话号码为:%@",telephone);

return NO;

}

return YES;

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,108评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,699评论 1 296
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,812评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,236评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,583评论 3 288
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,739评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,957评论 2 315
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,704评论 0 204
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,447评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,643评论 2 249
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,133评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,486评论 3 256
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,151评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,108评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,889评论 0 197
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,782评论 2 277
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,681评论 2 272

推荐阅读更多精彩内容