iOS 獲取 Label 高度的正確方式

orton 8年前發布 | 20K 次閱讀 iOS開發 移動開發

在設置 UILabel 的 Frame 高度時,不能簡單的設置為字體的 font size 。否則會將字體的一部分裁剪掉。因為 UILabel 在不同的字體設置下,對 Frame 的高度要求也不一樣,大多數情況下都比Font的高度設置要高一些。

sizeThatFits

使用 view 的 sizeThatFits 方法。

// return 'best' size to fit given size. does not actually resize view. Default is return existing view size
- (CGSize)sizeThatFits:(CGSize)size;

例子:

UILabel *testLabel = [[UILabel alloc] init];
testLabel.font = [UIFont systemFontOfSize:30];
testLabel.text = @"Today is a fine day";
CGSize size = [testLabel sizeThatFits:CGSizeMake(200, 30)];
NSLog(@"size = %@", NSStringFromCGSize(size));

輸出:size = {246.33333333333334, 36}

sizeToFit

使用 view 的 sizeToFit 方法。注意: sizeToFit 會改變 view 原來的 bounds ,而 sizeThatFits 不會。

// calls sizeThatFits: with current view bounds and changes bounds size.
- (void)sizeToFit;

例子

UILabel *testLabel = [[UILabel alloc] init];
testLabel.font = [UIFont systemFontOfSize:30];
testLabel.text = @"Today is a fine day";
[testLabel sizeToFit];
NSLog(@"size = %@", NSStringFromCGSize(testLabel.frame.size));

輸出:size = {246.33333333333334, 36}

sizeWithAttributes

使用 NSString 的 sizeWithAttributes 方法。

- (CGSize)sizeWithAttributes:(nullable NSDictionary<NSString *, id> *)attrs NS_AVAILABLE(10_0, 7_0);

例子

NSString *text = @"Today is a fine day";
UIFont *font = [UIFont systemFontOfSize:30];
CGSize size = [text sizeWithAttributes:@{
                                         NSFontAttributeName : font
                                         }];
NSLog(@"size = %@", NSStringFromCGSize(size));

輸出: size = {246.3134765625, 35.80078125}

boundingRectWithSize

使用 NSString 的 boundingRectWithSize 方法。

// NOTE: All of the following methods will default to drawing on a baseline, limiting drawing to a single line.
// To correctly draw and size multi-line text, pass NSStringDrawingUsesLineFragmentOrigin in the options parameter.
- (CGRect)boundingRectWithSize:(CGSize)size options:(NSStringDrawingOptions)options attributes:(nullable NSDictionary<NSString *, id> *)attributes context:(nullable NSStringDrawingContext *)context NS_AVAILABLE(10_11, 7_0);

參數的意義:

  1. size

    限制最大寬高, 雖然是自適應,但是需要限制最大的寬度和高度。

  2. options

    類型為 NSStringDrawingOptions ,用來指明繪制字符串時的渲染選項。各個選項如下:

    typedef NS_OPTIONS(NSInteger, NSStringDrawingOptions) {
      // The specified origin is the line fragment origin, not the base line origin
      // 整個文本將以每行組成的矩形為單位計算整個文本的尺寸
      NSStringDrawingUsesLineFragmentOrigin = 1 << 0, 
    
      // Uses the font leading for calculating line heights
      // 使用字體的行間距來計算文本占用的范圍,即每一行的底部到下一行的底部的距離計算
      NSStringDrawingUsesFontLeading = 1 << 1, 
    
      // Uses image glyph bounds instead of typographic bounds
      // 將文字以圖像符號計算文本占用范圍,而不是排版的邊界
      NSStringDrawingUsesDeviceMetrics = 1 << 3,
    
      // Truncates and adds the ellipsis character to the last visible line if the text doesn't fit into the bounds specified.
      // Ignored if NSStringDrawingUsesLineFragmentOrigin is not also set.
      // 如果文本內容超出指定的矩形限制,文本將被截去并在最后一個字符后加上省略號。
      // 如果 NSStringDrawingUsesLineFragmentOrigin 沒有設置,則該選項不生效
      NSStringDrawingTruncatesLastVisibleLine NS_ENUM_AVAILABLE(10_5, 6_0) = 1 << 5, 
    } NS_ENUM_AVAILABLE(10_0, 6_0);
  3. attributes

    應用于字符串的文本屬性。

  4. context

    NSStringDrawingContext 類型,控制調整字間距和縮放的比例,用于文本繪制時使用。該參數傳入 nil 即可。

例子

NSString *text = @"Today is a fine day";
UIFont *font = [UIFont systemFontOfSize:30];
CGRect suggestedRect = [text boundingRectWithSize:CGSizeMake(800, MAXFLOAT)
                                              options:NSStringDrawingUsesFontLeading
                                           attributes:@{ NSFontAttributeName : font }
                                              context:nil];
NSLog(@"size = %@", NSStringFromCGSize(suggestedRect.size));

輸出: size = {200, 35.80078125}

四種方式對比

在設置字體為 30 的情況下,前兩種使用 view 的方法返回 size = {246.33333333333334, 36},后兩種使用 NSString 的方法返回 size = {246.3134765625, 35.80078125}。使用 view 方法比使用 NSString 方法的返回的值略大。

我猜測其原因都是因為,文本渲染引擎在渲染一行文本的時候都需要在label的頂部和底部預留一小部分空間,應該是出于排版美觀方面的考量。

在顯示不同的 font size 的字體時,獲得的字符串高度比 font size 大的值是不同的。

比如 font size 為 13 時,算出高度為 16, font size 為 20 時,算出高度為 24。

所以平常設置 UILabel 高度的時候,也不能簡單的在 font height 基礎之上加隨意值。

參考

理解NSStringDrawingOptions每個選項的用法與意義

UILabel中一個很多人都會踩的坑

 

來自:http://www.jianshu.com/p/9278c2fbbe64

 

 本文由用戶 orton 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!