6

我正在将我的应用程序更新到 iOS 7 并最终得到它,但有一件事我找不到解决方案。

在 Xcode 4 中,我使用了以下方法:

#define FONT_SIZE 14.0f
#define CELL_CONTENT_WIDTH 280.0f
#define CELL_CONTENT_MARGIN 10.0f


- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath; {
    NSString *text = [textA objectAtIndex:[indexPath row]];

    CGSize constraint = CGSizeMake(CELL_CONTENT_WIDTH - (CELL_CONTENT_MARGIN * 2), 20000.0f);

    CGSize size = [text sizeWithFont:[UIFont systemFontOfSize:FONT_SIZE] constrainedToSize:constraint lineBreakMode:NSLineBreakByWordWrapping];

    CGFloat height = MAX(size.height, 28.0f);

    return height + (CELL_CONTENT_MARGIN * 2);
}

但在 iOS 7 中使用时会报错:

使用 -boundingRectWithSize:options:attributes:context:

我不知道如何将我的早期版本转换为这种新方法,如果有人可以帮助我,那就太好了。提前致谢。

4

3 回答 3

5

在 iOS7 中不推荐使用sizeWithFont方法。您应该改用boundingRectWithSize。如果您还需要支持之前的 iOS 版本,则可以使用以下代码:

CGSize size = CGSizeZero;

if ([label.text respondsToSelector: @selector(boundingRectWithSize:options:attributes:context:)] == YES) {
    size = [label.text boundingRectWithSize: constrainedSize options: NSStringDrawingUsesLineFragmentOrigin
                                 attributes: @{ NSFontAttributeName: label.font } context: nil].size;
} else {
    size = [label.text sizeWithFont: label.font constrainedToSize: constrainedSize lineBreakMode: UILineBreakModeWordWrap];
}
于 2013-09-18T19:06:18.937 回答
4

如果您只支持 ios6 及更高版本,则可以将您的转换NSStringsNSAttributedStrings并使用NSAttributedString's boundingRectWithSize:options:context:

以前看起来像这样的东西:

CGSize size = [text sizeWithFont:font
               constrainedToSize:(CGSize){maxWidth, CGFLOAT_MAX}];

可以很容易地转换为此并在 ios6 和 ios7 中工作:

NSAttributedString *attributedText =
    [[NSAttributedString alloc]
        initWithString:text
        attributes:@
        {
            NSFontAttributeName:    font
        }];
CGRect rect = [attributedText boundingRectWithSize:(CGSize){maxWidth, CGFLOAT_MAX}
                                           options:NSStringDrawingUsesLineFragmentOrigin
                                           context:nil];
CGSize size = rect.size;

附带说明一下,这样做的好处是,您在 ios6 中的文本大小现在是线程安全的。旧的方法sizeWithFont:...属于 UIStringDrawing,如果sizeWithFont:...同时运行在两个线程上会崩溃。在 ios6 中,NSAttributedStrings公开了新的 NSStringDrawing 函数,并且该boundingRectWithSize:...函数是线程安全的。我猜这就是为什么在 ios7sizeWithFont:...中不推荐使用旧功能的原因。

请注意文档中提到:

在 iOS 7 及更高版本中,此方法返回小数大小(在返回的 CGRect 的大小组件中);要使用返回的大小来调整视图大小,您必须使用 ceil 函数将其值提高到最接近的更高整数。

因此,要提取用于调整视图大小的计算高度或宽度,我会使用:

CGFloat height = ceilf(size.height);
CGFloat width  = ceilf(size.width);
于 2013-09-22T15:05:40.500 回答
2

您正在使用的 sizeWithFont API 在 iOS7 上已弃用。

// See UIStringDrawing.h
- (CGSize)sizeWithFont:(UIFont *)font constrainedToSize:(CGSize)size lineBreakMode:(NSLineBreakMode)lineBreakMode NS_DEPRECATED_IOS(2_0, 7_0, "Use -boundingRectWithSize:options:attributes:context:"); // NSTextAlignment is not needed to determine size

您可以像这样使用 API 建议:

NSMutableDictionary *atts = [[NSMutableDictionary alloc] init];
[atts setObject:myFont forKey:NSFontAttributeName];

CGRect rect = [myText boundingRectWithSize:constraint
                                 options:NSStringDrawingUsesLineFragmentOrigin
                              attributes:atts
                                 context:nil];
于 2013-09-16T18:49:53.853 回答