因为某些原因,不便使用Auto Layout或者需要计算text高度确定view高度,有同学遇到UITextView高度计算错误,导致显示错误的问题。
用过UITextView的同学都知道,UITextView的text默认是内嵌的,也就是它的内容默认不是铺满整个view。代码和效果图(图一)如下:
UITextView *textView = [[UITextView alloc] initWithFrame:CGRectZero];
textView.center = self.view.center;
textView.font = [UIFont systemFontOfSize:16.f];
textView.scrollEnabled = NO;
NSString *text = @"qwwerqwerqwerqwerqweqwerqwerqwerqwerqwerqwerqwerqwerqweqwerqwerqwerqwerqwerqw";
textView.text = text;
CGFloat height = [text boundingRectWithSize:CGSizeMake(200, CGFLOAT_MAX) options:NSStringDrawingUsesFontLeading | NSStringDrawingUsesLineFragmentOrigin attributes:@{NSFontAttributeName:textView.font} context:nil].size.height;
textView.bounds = CGRectMake(0, 0, 200, height);
[self.view addSubview:textView];
图一可以看出它的上下左右是有边距的。所以你按照它的宽度去计算文本的高度是错误的。
iOS7 之后有了 textContainerInset 这个属性。
ok,我们设置 textView.textContainerInset = UIEdgeInsetsZero;这句话的理解很多同学认为是把textView的边距设为0,那我们看一下效果图:
图二和图一相比,这也不是我们想象中的结果,官方解释如图三
大致意思是内嵌文本容器的布局区域在view的内容区域内,好像是废话。
图三其实,textView还有一个属性:textContainer,通过它可以拿到内嵌文本容器在textView里的padding。
NSLog(@"\n\
padding = %lf\n\
textContainerInsetTop = %lf \n\
textContainerInsetLeft = %lf \n\
textContainerInsetBottom = %lf \n\
textContainerInsetRight = %lf \n",textView.textContainer.lineFragmentPadding, textView.textContainerInset.top, textView.textContainerInset.left, textView.textContainerInset.bottom, textView.textContainerInset.right);
输出看图四:
图四所以,只设置textView.textContainerInset = UIEdgeInsetsZero,是把文本在内嵌容器的边距设为0。
计算高度的时候这样计算,方法一:
CGFloat height = [text boundingRectWithSize:CGSizeMake(200 - textView.textContainer.lineFragmentPadding * 2, CGFLOAT_MAX) options:NSStringDrawingUsesFontLeading | NSStringDrawingUsesLineFragmentOrigin attributes:@{NSFontAttributeName:textView.font} context:nil].size.height;//宽度减去左右两边的padding
textView.bounds = CGRectMake(0, 0, 200, height);
或者设置,方法二 :
CGFloat padding = textView.textContainer.lineFragmentPadding;
textView.textContainerInset = UIEdgeInsetsMake(0, -padding, 0, -padding);
这个时候计算的高度都是正确的,方法二的效果图:
图五
网友评论