Подтвердить что ты не робот

Динамический расчет ширины UILabel в UITableViewCell

Я создаю пользовательское представление настроек в приложении, основанное на UITableViewCellStyle1 (или так).

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

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

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

- (UITableViewCell*)tableView:(UITableView*)tableView cellForRowAtIndexPath:(NSIndexPath*)indexPath
  {
      static NSString* reuseIdentifier = @"SettingsCell";

UITableViewCell* cell = [tableView dequeueReusableCellWithIdentifier:reuseIdentifier];

if ( !cell )
{   
    cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:reuseIdentifier] autorelease];
    [cell setBackgroundColor:[UIColor baeDarkGrayColor]];


    UILabel* cellLabel = [cell textLabel];
    [cellLabel setTextColor:[UIColor whiteColor]];
    [cellLabel setBackgroundColor:[UIColor clearColor]];

    [cell setUserInteractionEnabled:YES];
    [cell setSelectionStyle:UITableViewCellSelectionStyleGray];
}

// Populate cell with corresponding data.
NSDictionary* tableSection = [_tableData objectAtIndex:indexPath.section];

// Set the label
UILabel* textLabel = [cell textLabel];
NSString* labelString = [[tableSection objectForKey:@"itemTitles"] objectAtIndex:indexPath.row];
[textLabel setText:labelString];

   //   CGSize constrainedSize;
  //    constrainedSize.width = MAXFLOAT;
 // constrainedSize.height = MAXFLOAT;

CGSize textLabelSize = [textLabel.text sizeWithFont:textLabel.font /*constrainedToSize:constrainedSize*/];

NSLog(@"text label width: %f", textLabelSize.width);    


// Set the accessory view
BAESettingsTableCellAccessoryType accessoryType = [[[tableSection objectForKey:@"itemAccessories"] objectAtIndex:indexPath.row] integerValue];
switch ( accessoryType )
{
    case BAESettingsTableCellAccessoryDisclosureIndicator:
    {
        UIImageView* disclosureView = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"AccessoryDisclosure.png"]];
        [cell setAccessoryView:disclosureView];
        [disclosureView release];
        break;
    }
    case BAESettingsTableCellAccessoryOnOffSwitch:
    {
        UISwitch* onOffSwitch = [[UISwitch alloc] initWithFrame:CGRectZero];
        [onOffSwitch setOn:YES];
        [cell setAccessoryView:onOffSwitch];
        [onOffSwitch release];
        break;
    }
    case BAESettingsTableCellAccessoryNone:
        // default:
        // break;
    {

        CGRect cellFrame = [cell frame];

        float detailTextFieldWidth = cellFrame.size.width - ( textLabelSize.width + 50 );
        float detailTextFieldHeight = cellFrame.size.height - 1;

        NSLog(@"detail text field calculated frame width, height, %f, %f", detailTextFieldWidth, detailTextFieldHeight);


        CGRect frame = CGRectMake(cellFrame.origin.x, cellFrame.origin.y, detailTextFieldWidth, detailTextFieldHeight);


        UITextField* textField = [[UITextField alloc] initWithFrame:frame];
        NSString* detailLabelString = [[tableSection objectForKey:@"itemDetailStrings"] objectAtIndex:indexPath.row];
        textField.text = detailLabelString;
        textField.textColor = cell.detailTextLabel.textColor;
        textField.textAlignment = UITextAlignmentRight;
        textField.borderStyle = UITextBorderStyleRoundedRect;
        [cell setAccessoryView:textField];
        [textField release];
        break;
    }

}

return cell;
   }
4b9b3361

Ответ 1

Явная настройка шрифта решает вашу проблему.

CGSize textLabelSize = [textLabel.text sizeWithFont:[UIFont systemFontOfSize:17]];

Горы:
Когда tableview сначала загружается, этот фрагмент возвращает нулевую ширину из-за того, что размер шрифта равен 0.

[textLabel.text sizeWithFont:textLabel.font]

Я понял это, указав размер точки шрифта ячейки.

NSLog(@"font size: %f", cell.textLabel.font.pointSize);

Размер точки был равен нулю, пока ячейка не погаснет и не вернется, как вы описали.

Ответ 2

Я полностью согласен с тем, что это достаточно раздражает, что это должно быть ошибка. Почему приятные люди в Apple когда-либо просят высоты клетки, не гарантируя, что ячейка полностью настроена. Глупо! Однако вместо использования устаревшей функции попробуйте:

[cell layoutIfNeeded];

Вызов этого метода имеет тот же эффект с настройкой ячейки, включая границы, шрифт ячейки и т.д. Поэтому любой расчет после этого имеет доступ к полностью настроенной ячейке. Здесь весь код:

-(CGFloat) tableView:(UITableView *) tableView heightForRowAtIndexPath:(NSIndexPath *) indexPath {

  // Find the cell for this index path
  UITableViewCell *cell = [self tableView:tableView cellForRowAtIndexPath:indexPath];
  CGFloat cellHeight = cell.frame.size.height;

  // Calculate text size after forcing a layout
  [cell layoutIfNeeded];
  CGSize textSize = [cell.textLabel.text sizeWithFont:cell.textLabel.font constrainedToSize:CGSizeMake(cell.contentView.bounds.size.width, 99999) lineBreakMode:cell.textLabel.lineBreakMode];

  // Only change the cell height if the text forces a growth
  if (textSize.height > cellHeight) {
    cellHeight = textSize.height;
  }

  return cellHeight;
}

Ответ 3

Я нашел ту же проблему, но я нашел обходное решение. FYI Я поднял ошибку с Apple (ошибка № 7535066). Лекс, в вашем примере, если вы используете cell.font вместо textLabel.font, он должен работать, хотя вы получите предупреждение компилятора, так как это вызов устаревшего метода. Я не уверен, что вы сможете увидеть подробности об ошибке, которую я поднял, поэтому я вставляю детали здесь:

iPhone SDK 3.1.2 (7D11) NSString UIKit Метод добавления sizeWithFont возвращает nil

РЕЗЮМЕ Я украшаю UITableViewCells в методе делегата UITableView:

  • (UITableViewCell *) tableView: (UITableView *) tableView cellForRowAtIndexPath: (NSIndexPath *) indexPath

Я хочу найти длину текста уже в ячейке, чтобы правильно форматировать новое представление, поэтому я вызываю метод NCCtring UIKit Additions:

  • (CGSize) sizeWithFont: (UIFont *) font constrainedToSize: (CGSize) размер

Когда я использую cell.textLabel.font для аргумента шрифта, он всегда возвращает пустой CGSize

ЕСЛИ я использую метод DEPRECATED, cell.font для аргумента, он возвращает значение CGSize, как ожидалось.

ШАГИ ДЛЯ ВОСПРОИЗВЕДЕНИЯ 1. Создайте UITableViewController с TableView и запишите его примерными данными 2. В заштрихованном методе cellForRowAtIndexPath после комментария//Настройте ячейку и где заполняется ячейка .textLabel.text, поставьте следующее:

  
 CGSize size = [cell.textLabel.text sizeWithFont:[[cell textLabel] font] constrainedToSize:CGSizeMake(320, 480)];
 NSLog(@"cell frame: %f %f", size.width, size.height);

 CGSize size2 = [cell.textLabel.text sizeWithFont:[cell font] constrainedToSize:CGSizeMake(320, 480)];
 NSLog(@"DEPRECATED cell frame: %f %f", size2.width, size2.height);

ОЖИДАЕМЫЕ РЕЗУЛЬТАТЫ

В любом случае вы ожидаете, что вызов, который использует метод "Устаревший", для получения шрифта, для отказа или, по крайней мере, для того, чтобы вернуть тот же результат.

АКТУАЛЬНЫЙ РЕЗУЛЬТАТ:

2010-01-12 22: 06: 36.645 PrefTest [9659: 207] ячейка: 0,000000 0,000000 2010-01-12 22: 06: 36.646 PrefTest [9659: 207] DEPRECATED рама ячейки: 88.000000 24.000000

ПРОГРАММИРОВАНИЕ И ИЗОЛЯЦИЯ:

Как отмечено, используйте устаревший метод для получения шрифта. Но это становится страннее!

Переверните два метода, которые я дал выше:

 
 CGSize size2 = [cell.textLabel.text sizeWithFont:[cell font] constrainedToSize:CGSizeMake(320, 480)];
 NSLog(@"DEPRECATED cell frame: %f %f", size2.width, size2.height);

    CGSize size = [cell.textLabel.text sizeWithFont:[[cell textLabel] font] constrainedToSize:CGSizeMake(320, 480)];
 NSLog(@"cell frame: %f %f", size.width, size.height);

Теперь результаты:

2010-01-12 22: 06: 36.645 PrefTest [9659: 207] ячейка: 88.000000 24.000000 2010-01-12 22: 06: 36.646 PrefTest [9659: 207] DEPRECATED рама ячейки: 88.000000 24.000000

Похоже, что для ссылки на [ячейка шрифт] достаточно, чтобы [[text TextLabel]] правильно работал.

Ответ 4

Используйте [myLbl sizeToFit];, чтобы автоматически изменить размер метки.

Вот пример кода.

UILabel *lbl1, *lbl2;
    if (cell == nil) {

   UILabel *lbl1 = [[UILabel alloc] init];
            [lbl1 setFont:[UIFont boldSystemFontOfSize:20]];
            [cell.contentView addSubview:lbl1];
            lbl1.frame = CGRectMake(3, 10, 0, 0);
            lbl1.tag = 10;
            [lbl1 release];

        UILabel *lbl2 = [[UILabel alloc] init];
        [lbl2 setFont:[UIFont systemFontOfSize:20]];
        [cell.contentView addSubview:lbl2];                 
        lbl2.tag = 20;
        [lbl2 release];
}
else {
       lbl1 = (UILabel*)[cell.contentView viewWithTag:10];
       lbl2 = (UILabel*)[cell.contentView viewWithTag:20];
}

  lbl1.text = @"Hello ";
    [lbl1 sizeToFit];

lbl2.text = @"World";
        [lbl2 sizeToFit];
        lbl2.frame = CGRectMake(lbl1.frame.origin.x+lbl1.frame.size.width+5,
                                lbl1.frame.origin.y,
                                lbl2.frame.size.width,
                                lbl1.frame.size.height);