Как найти UILabel количество строк

Я отобразил текст в UILabel с помощью метода wrap. Теперь мне нужно найти количество строк в UILabel.

Если есть возможный способ найти количество строк в строке UILabel.

Спасибо.

Ответ 1

Как указано, этот пост описывает, как получить высоту, а не количество строк. Чтобы получить количество строк,

  • Получить высоту для одной буквы, например. @"A".
  • Разделите высоту строки на высоту, полученную в 1 выше.

например.

CGFloat unitHeight = [@"A" heightForWidth:width usingFont:font];
CGFloat blockHeight = [text heightForWidth:width usingFont:font];
NSInteger numberOfLines =        ceilf(blockHeight / unitHeight); 
// need to #include <math.h> for ^^^^^

Начиная с iOS 7, изменился способ получения метки желаемой высоты. Чтобы получить высоту, вы можете использовать следующий код:

NSStringDrawingContext *context = [[NSStringDrawingContext alloc] init];
CGSize labelSize = (CGSize){width, FLT_MAX};
CGRect r = [self boundingRectWithSize:labelSize options:NSStringDrawingUsesLineFragmentOrigin attributes:@{NSFontAttributeName: font} context:context];

где высота r.size.height. Обратите внимание, что font должен быть предоставлен. Вы можете поместить это в категорию для NSString для удобства, например.

@implementation NSString (HeightCalc)

- (CGFloat)heightForWidth:(CGFloat)width usingFont:(UIFont *)font
{
    NSStringDrawingContext *context = [[NSStringDrawingContext alloc] init];
    CGSize labelSize = (CGSize){width, FLT_MAX};
    CGRect r = [self boundingRectWithSize:labelSize options:NSStringDrawingUsesLineFragmentOrigin attributes:@{NSFontAttributeName: font} context:context];
    return r.size.height;
}

@end

(Управление памятью, если не используется ARC, кстати.)

Для iOS 6 и ниже:

Скажем, у вас есть UILabel *myLabel, и вы хотите узнать высоту метки (с некоторой настройкой вы можете получить # строк, разделив высоту с некоторым соответствующим числом, которое зависит от размера шрифта).

UILabel *myLabel;
CGSize labelSize = [myLabel.text sizeWithFont:myLabel.font 
                            constrainedToSize:myLabel.frame.size 
                                lineBreakMode:UILineBreakModeWordWrap];
CGFloat labelHeight = labelSize.height;

Надеюсь, что это поможет. Если это не поможет, дайте мне знать, и я буду копать дальше. Кроме того, непроверенный код, но работал со ссылкой.

Для более полного примера, вот код, который я ввел в метод viewDidLoad: для контроллера вида:

[super viewDidLoad];
UILabel *myLabel = [[UILabel alloc] initWithFrame:CGRectMake(50,50,200,350)];
myLabel.numberOfLines = 0;
myLabel.lineBreakMode = UILineBreakModeWordWrap;
myLabel.text = @"This is some text in a UILabel which is long enough to wrap around the lines in said UILabel. This is a test, this is only a test.";
[self.view addSubview:myLabel];
CGSize labelSize = [myLabel.text sizeWithFont:myLabel.font 
                            constrainedToSize:myLabel.frame.size 
                                lineBreakMode:UILineBreakModeWordWrap];
CGFloat labelHeight = labelSize.height;
NSLog(@"labelHeight = %f", labelHeight);
[myLabel release];

Выход из NSLog идет:

2010-11-15 18:25:27.817 so_labelheight[728:307] labelHeight = 126.000000

Ответ 2

Для iOS7 и выше официально санкционированный способ для подсчета количества строк используется TextKit:

func numberOfLinesForString(string: String, size: CGSize, font: UIFont) -> Int {
    let textStorage = NSTextStorage(string: string, attributes: [NSFontAttributeName: font])

    let textContainer = NSTextContainer(size: size)
    textContainer.lineBreakMode = .ByWordWrapping
    textContainer.maximumNumberOfLines = 0
    textContainer.lineFragmentPadding = 0

    let layoutManager = NSLayoutManager()
    layoutManager.textStorage = textStorage
    layoutManager.addTextContainer(textContainer)

    var numberOfLines = 0
    var index = 0
    var lineRange : NSRange = NSMakeRange(0, 0)
    for (; index < layoutManager.numberOfGlyphs; numberOfLines++) {
        layoutManager.lineFragmentRectForGlyphAtIndex(index, effectiveRange: &lineRange)
        index = NSMaxRange(lineRange)
    }

    return numberOfLines
}

Ответ 3

Метод -sizeWithFont:constrainedToSize:lineBreakMode теперь устарел. Теперь вы захотите использовать метод -boundingRectWithSize:options:attributes:context:.

Вот примеры:

CGSize boundingRectSize = CGSizeMake(widthToConstrainTo, CGFLOAT_MAX);
NSDictionary *attributes = @{NSFontAttributeName : [UIFont fontWithName:fontName size:14]};
CGRect labelSize = [labelString boundingRectWithSize:boundingRectSize options:NSStringDrawingUsesLineFragmentOrigin | NSStringDrawingUsesFontLeading
                                                attributes:attributes
                                                   context:nil];

В приведенном выше примере я знаю ширину, которую я хочу ограничить меткой, но поскольку я не уверен в высоте, я просто увеличиваю параметр высоты с помощью CGFLOAT_MAX. Для options вам нужно использовать NSStringDrawingUsesLineFragmentOrigin и NSStringDrawingUsesFontLeading, если вы пытаетесь вычислить размер метки, которая может быть любым количеством строк.

Ответ 4

sizeWithFont устарел в iOS 7, вы можете использовать это вместо:

- (int)lineCountForText:(NSString *) text
{
    UIFont *font = ...

    CGRect rect = [text boundingRectWithSize:CGSizeMake(200, MAXFLOAT)
                                     options:NSStringDrawingUsesLineFragmentOrigin
                                  attributes:@{NSFontAttributeName : font}
                                     context:nil];

    return ceil(rect.size.height / font.lineHeight);
}

Ответ 5

Если вы ищете этот ответ в Swift 2 +/iOS 8, он будет выглядеть следующим образом:

func numberOfLinesInLabel(yourString: String, labelWidth: CGFloat, labelHeight: CGFloat, font: UIFont) -> Int {

    let paragraphStyle = NSMutableParagraphStyle()
    paragraphStyle.minimumLineHeight = labelHeight
    paragraphStyle.maximumLineHeight = labelHeight
    paragraphStyle.lineBreakMode = .ByWordWrapping

    let attributes: [String: AnyObject] = [NSFontAttributeName: font, NSParagraphStyleAttributeName: paragraphStyle]

    let constrain = CGSizeMake(labelWidth, CGFloat(Float.infinity))

    let size = yourString.sizeWithAttributes(attributes)
    let stringWidth = size.width

    let numberOfLines = ceil(Double(stringWidth/constrain.width))

    return Int(numberOfLines)
}

Ответ 6

(Я изначально отправил свой ответ здесь, но не имеет достаточной репутации для комментариев или отметки дубликатов.)

Другие ответы, которые я нашел, не уважают свойство numberOfLines UILabel, если для него установлено значение, отличное от 0.

Вот еще один вариант, который вы можете добавить в свою категорию или подкласс:

- (NSUInteger)lineCount
{
    CGSize size = [self sizeThatFits:CGSizeMake(self.frame.size.width, CGFLOAT_MAX)];
    return MAX((int)(size.height / self.font.lineHeight), 0);
}

Некоторые примечания:

  • Я использую это в UILabel с приписанным текстом, даже не устанавливая свойство font, и он работает нормально. Очевидно, что вы столкнулись с проблемами, если бы вы использовали несколько шрифтов в attributedText.
  • Если вы выполняете подклассификацию UILabel, чтобы иметь пользовательские вставки кромки (например, переопределяя drawTextInRect:, что является опрятным трюком, я нашел здесь), тогда вы не должны учитывать эти вставки при расчете size выше. Например: CGSizeMake(self.frame.size.width - (self.insets.left + self.insets.right), CGFLOAT_MAX)

Ответ 7

func getHeight(text:  NSString, width:CGFloat, font: UIFont) -> CGFloat
{
    let rect = text.boundingRect(with: CGSize.init(width: width, height: CGFloat.greatestFiniteMagnitude), options: ([NSStringDrawingOptions.usesLineFragmentOrigin,NSStringDrawingOptions.usesFontLeading]), attributes: [NSFontAttributeName:font], context: nil)
    return rect.size.height
}

текст: метка, высота которой вам нужно найти с заданной строкой

width: ширина UILabel

font: шрифт UILabel