我正在将一个应用程序从iOS 6.1移植到iOS 7.我使用的是一个布局,其中UITextView
具有固定宽度,但它的高度基于其内容大小。对于iOS 6.1,检查contentsize.height并将其设置为textview的帧高度就足够了,但它在iOS 7上不起作用。
如何创建一个具有固定宽度的UITextView
,但是根据它显示的文字创建动态高度?
注意:我是从代码创建这些视图,而不是使用Interface Builder。
答案 0 :(得分:179)
使用以下代码,您可以根据固定宽度更改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];
}
}
您可以在项目的prefix.pch文件中添加SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO
:
#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:)])
答案 1 :(得分:42)
这适用于iOS6和7:
CGSize textViewSize = [self.myTextView sizeThatFits:CGSizeMake(self.myTextView.frame.size.width, FLT_MAX)];
self.myTextView.height = textViewSize.height;
答案 2 :(得分:20)
在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;
答案 3 :(得分:15)
使用这个小功能
-(CGSize) getContentSize:(UITextView*) myTextView{
return [myTextView sizeThatFits:CGSizeMake(myTextView.frame.size.width, FLT_MAX)];
}
答案 4 :(得分:3)
我的最终解决方案基于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);
}
答案 5 :(得分:2)
使用此方法有更简单的解决方案:
+(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)
答案 6 :(得分:1)
undefined method `user' for nil:NilClass
不要忘记lineFragmentPadding
答案 7 :(得分:0)
简单解决方案 - textView.isScrollEnabled = false
在另一个滚动视图或具有UITableViewAutomaticDimension
答案 8 :(得分:0)
@ Ana,@ Blake Hamilton在 swift 中的解决方案。
var contentHeight: CGFloat = textView.sizeThatFits(textView.frame.size).height
对我来说,好消息是,当contentSize
设置为false时,它还会返回正确的isScrollEnable
。设置为false将返回文本视图的框架大小,而不是内容大小。