我正在将其中一个应用程序从 iOS 6.1 移植到 iOS 7。我正在使用一个布局,其中有一个UITextView
具有固定宽度的布局,但它的高度取决于其内容大小。对于 iOS 6.1,检查 contentsize.height 并将其设置为 textview 的框架高度就足够了,但它在 iOS 7 上不起作用。
然后如何UITextView
根据它显示的文本创建具有固定宽度但动态高度的?
注意:我是从代码中创建这些视图,而不是使用 Interface Builder。
我正在将其中一个应用程序从 iOS 6.1 移植到 iOS 7。我正在使用一个布局,其中有一个UITextView
具有固定宽度的布局,但它的高度取决于其内容大小。对于 iOS 6.1,检查 contentsize.height 并将其设置为 textview 的框架高度就足够了,但它在 iOS 7 上不起作用。
然后如何UITextView
根据它显示的文本创建具有固定宽度但动态高度的?
注意:我是从代码中创建这些视图,而不是使用 Interface Builder。
使用以下代码,您可以根据固定宽度更改 UITextView 的高度(它适用于 iOS 7 和以前的版本):
- (CGFloat)textViewHeightForAttributedText:(NSAttributedString *)text andWidth:(CGFloat)width
{
UITextView *textView = [[UITextView alloc] init];
[textView setAttributedText:text];
CGSize size = [textView sizeThatFits:CGSizeMake(width, FLT_MAX)];
return size.height;
}
使用此函数,您将采用 NSAttributedString 和固定宽度来返回所需的高度。
如果要从具有特定字体的文本中计算框架,则需要使用以下代码:
- (CGSize)text:(NSString *)text sizeWithFont:(UIFont *)font constrainedToSize:(CGSize)size
{
if (SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(@"7.0"))
{
CGRect frame = [text boundingRectWithSize:size
options:(NSStringDrawingUsesLineFragmentOrigin | NSStringDrawingUsesFontLeading)
attributes:@{NSFontAttributeName:font}
context:nil];
return frame.size;
}
else
{
return [text sizeWithFont:font constrainedToSize:size];
}
}
您可以将其添加到SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO
项目中的 prefix.pch 文件中:
#define SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(v) ([[[UIDevice currentDevice] systemVersion] compare:v options:NSNumericSearch] != NSOrderedAscending)
您还可以将之前的测试替换为SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(v)
:
if ([text respondsToSelector:@selector(boundingRectWithSize:options:attributes:context:)])
这适用于 iOS6 和 7:
CGSize textViewSize = [self.myTextView sizeThatFits:CGSizeMake(self.myTextView.frame.size.width, FLT_MAX)];
self.myTextView.height = textViewSize.height;
在 iOS7 中,UITextView
用于NSLayoutManager
布局文本:
// If YES, then the layout manager may perform glyph generation and layout for a given portion of the text, without having glyphs or layout for preceding portions. The default is NO. Turning this setting on will significantly alter which portions of the text will have glyph generation or layout performed when a given generation-causing method is invoked. It also gives significant performance benefits, especially for large documents.
@property(NS_NONATOMIC_IOSONLY) BOOL allowsNonContiguousLayout;
禁用allowsNonContiguousLayout
修复contentSize
:
textView.layoutManager.allowsNonContiguousLayout = NO;
使用这个小功能
-(CGSize) getContentSize:(UITextView*) myTextView{
return [myTextView sizeThatFits:CGSizeMake(myTextView.frame.size.width, FLT_MAX)];
}
我的最终解决方案基于 HotJard,但包括文本容器的顶部和底部插图,而不是使用 2*fabs(textView.contentInset.top) :
- (CGFloat)textViewHeight:(UITextView *)textView
{
return ceilf([textView.layoutManager usedRectForTextContainer:textView.textContainer].size.height +
textView.textContainerInset.top +
textView.textContainerInset.bottom);
}
有更简单的解决方案,使用这种方法:
+(void)adjustTextViewHeightToContent:(UITextView *)textView;
{
if([[UIDevice currentDevice].systemVersion floatValue] >= 7.0f){
textView.height = [textView.layoutManager usedRectForTextContainer:textView.textContainer].size.height+2*fabs(textView.contentInset.top);
}else{
textView.height = textView.contentSize.height;
}
}
UPD:仅用于显示文本(isEditable = NO)
_textView.textContainerInset = UIEdgeInsetsZero;
_textView.textContainer.lineFragmentPadding = 0;
只是不要忘记 lineFragmentPadding
简单的解决方案 -textView.isScrollEnabled = false
在另一个滚动视图或 tableView 单元格中时完美工作UITableViewAutomaticDimension
@Ana's,@Blake Hamilton 在swift中的解决方案。
var contentHeight: CGFloat = textView.sizeThatFits(textView.frame.size).height
对我来说好处是contentSize
,当isScrollEnable
设置为 false 时,它也会返回正确的。设置为 false 返回文本视图的框架大小而不是内容大小。