计算Cell高度一直是一个很热的问题,在IM app大量涌现之后,这问题就更加频繁了。我这里说一下计算NSAttributedString高度的方法,纯代码。
首先,普通的文本sizetofit 就好了,所以不存在难度。那么图文混排呢?一般人会说用CoreText,不过你用了就知道了,谁用谁傻。iOS7 开始Apple提供了TextKit来处理图文混排。这个的方法比较简单,而且直观。
TextKit实现图文混排
我先贴一下code,然后慢慢解释一下。
第一步,拼接string
1 /*! 2 * 插入表情 3 * 4 * @param rangeArray 所要插入表情的位置 每个元素包括(loc) 5 * @param facesArray 要插入的表情的信息 每个model包括(id imageName) 6 * @param pStr 原始的字符串 7 * @param sourceArray 表情包 8 * 9 * @return 拼接好的字符串 10 */ 11 + (NSMutableAttributedString *)setupEmojiByRangeArray:(NSArray *)rangeArray facesArray:(NSArray *)facesArray plainStr:(NSAttributedString *)pStr sourceArray:(NSArray *)sourceArray { 12 NSMutableAttributedString * mutStr = [pStr mutableCopy]; 13 __block int offset = 0;//每次加过一个表情后 字符串会变长 这个变量会记录已经增加的长度,不然插入的位置会错位的 14 if(rangeArray.count == facesArray.count) { // 看看要插入的位置的数量 是不是和要插入的表情数量一致 15 [facesArray enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {// 数组的枚举 16 for(FaceModel * fm in sourceArray){// 这个在不在表情包里面 17 if(fm.id == [obj[@"ID"]intValue]) {
18 UIImage * image1 = [UIImage imageNamed:fm.imageName]; 19 NSTextAttachment * attachment1 = [[NSTextAttachment alloc] init];//这就是要插入的对象 20 attachment1.bounds = CGRectMake(0, -3, 20, 20);// 插入的表情的大小 21 attachment1.image = image1; 22 NSAttributedString * attachStr1 = [NSAttributedString attributedStringWithAttachment:attachment1];// 把要插入的对象转为string 23 NSDictionary * dic = [rangeArray objectAtIndex:idx];// 看看插入哪个位置 24 [mutStr insertAttributedString:attachStr1 atIndex:offset+[dic[@"loc"]intValue]]; 25 offset += (int)attachStr1.length;// 插入结束要计算一下偏移量 26 } 27 } 28 }]; 29 } 30 // //添加链接 31 // NSURL * url = [NSURL URLWithString:@"http://www.baidu.com"]; 32 // [mutStr addAttribute:NSLinkAttributeName value:url range:NSMakeRange(loc, length)]; 33 // mainText.attributedText = [mutStr copy]; 34 35 return mutts;// 返回处理好的字符串 36 }
仔细看一下上面的代码,我做了表情的插入。这个是IM的必要需求了。通过NSTextAttachment的使用,我们可以插入表情的image了。因为我这个是批量处理,所以我插入规格一致的表情。同理,我们可以插入一张图片。。。对,就是图片,这就图文混排了。
第二步,计算NSMutableAttributedString的Rect
1 // 字体大小 2 UIFont *font =[UIFont systemFontOfSize:14];
3 // 段落设置 4 NSMutableParagraphStyle *p = [[NSMutableParagraphStyle alloc]init]; 5 p.lineBreakMode = NSLineBreakByCharWrapping; 6 p.lineSpacing = 0.1f; 7 p.paragraphSpacing = 0.1f; 8 p.alignment = NSTextAlignmentLeft; 9 // 设置NSAttributedString 的样式 10 NSAttributedString * str = [[NSAttributedString alloc]initWithString:rString attributes:@{NSFontAttributeName:font,NSParagraphStyleAttributeName:p}]; 11 // 计算Rect 12 // w代表你想要的宽度 h代表你的字符串最大支持的高度 13 CGRect rect = [sstr boundingRectWithSize:CGSizeMake(w,h) options:NSStringDrawingUsesFontLeading|NSStringDrawingUsesLineFragmentOrigin context:nil];
依靠计算出来的rect 我们就知道要怎么设置cell的高度了:这个string放到一个textView里面,textvView放到cell上。一切OK。。。