Настройка стиля UITableViewCell при использовании iOS 6 UITableView dequeueReusableCellWithIdentifier: forIndexPath:

Я пытаюсь определить, как установить UITableViewCellStyle при использовании новых методов в iOS 6 для UITableView.

Раньше при создании UITableViewCell я бы изменил перечисление UITableViewCellStyle для создания различных типов ячеек по умолчанию при вызове initWithStyle:, но из того, что я могу собрать, это уже не так.

Документация Apple для UITableView гласит:

Возвращаемое значение: Объект UITableViewCell с ассоциированным идентификатором повторного использования. Этот метод всегда возвращает действительную ячейку.

Обсуждение: По соображениям производительности источник данных табличного представления обычно должен повторно использовать объекты UITableViewCell, когда он назначает ячейки строкам в своем методе tableView: cellForRowAtIndexPath:. В представлении таблицы поддерживается очередь или список объектов UITableViewCell, которые источник данных отмечен для повторного использования. Вызовите этот метод из объекта источника данных, когда его попросят предоставить новую ячейку для представления таблицы. Этот метод деактивирует существующую ячейку, если она доступна, или создает новую, основанную на ранее сохраненном файле класса или файла nib.

Важно. Вы должны зарегистрировать файл класса или nib, используя метод registerNib: forCellReuseIdentifier: или registerClass: forCellReuseIdentifier: перед вызовом этого метода.

Если вы зарегистрировали класс для указанного идентификатора и создать новую ячейку, этот метод инициализирует ячейку, вызывая ее метод initWithStyle: reuseIdentifier:. Для ячеек, основанных на nib, этот метод загружает объект ячейки из предоставленного файла nib. Если существующая ячейка была доступна для повторного использования, этот метод вместо этого вызывает метод prepareForReuse.

Вот как выглядит мой новый cellForRowAtIndexPath после реализации новых методов:

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

    [tableView registerClass:[UITableViewCell class] forCellReuseIdentifier:cellIdentifier];

    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier forIndexPath:indexPath];

    return cell;
}

Код, который я до сих пор работает нормально, но всегда возвращает стиль по умолчанию. Как я могу изменить это, чтобы я мог создавать ячейки с другими стилями, такими как UITableViewCellStyleDefault, UITableViewCellStyleValue1, UITableViewCellStyleValue2 и UITableViewCellStyleSubtitle?

Я не хочу подкласса UITableViewCell, я просто хочу изменить тип по умолчанию, как я мог бы сделать до iOS 6. Кажется странным, что Apple предоставит расширенные методы, но с минимальной документацией для поддержки их реализации.

Кто-нибудь справился с этим или столкнулся с подобной проблемой? Я изо всех сил стараюсь найти любую разумную информацию.

Ответ 1

Я знаю, что вы сказали, что не хотите создавать подкласс, но он выглядит неизбежным. На основе кода сборки во время тестирования в симуляторе iOS 6.0 UITableView создает новые экземпляры UITableViewCell (или его подклассы), выполняя

[[<RegisteredClass> alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:<ReuseIdentifier>]

Другими словами, стиль, отправленный (UITableViewCellStyleDefault), выглядит жестко. Чтобы обойти это, вам нужно создать подкласс, который переопределяет инициализатор по умолчанию initWithStyle:reuseIdentifier: и передает стиль, который вы хотите использовать:

- (id)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier
{
    // ignore the style argument, use our own to override
    self = [super initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:reuseIdentifier];
    if (self) {
        // If you need any further customization
    }
    return self;
}

Кроме того, лучше было бы отправить registerClass:forCellReuseIdentifier: в viewDidLoad вместо того, чтобы делать это каждый раз, когда запрашивается ячейка:

- (void)viewDidLoad
{
    [super viewDidLoad];
    [self.tableView registerClass:<RegisteredClass> forCellReuseIdentifier:<ReuseIdentifier>];
}

Ответ 2

dequeueReusableCellWithIdentifier не устарел, поэтому вам не нужно использовать новый dequeueReusableCellWithIdentifier:forIndexPath:.

Используйте новый способ вместе с соответствующим методом register (в viewDidLoad), если вы используете собственный класс ячеек, но используете старый способ, если вы хотите использовать одну из перечислений UITableViewCellStyle.

Ответ 3

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

  • В представлении "Раскадровка" выберите ячейку прототипа ячейки таблицы (в представлении таблицы).
  • В представлении "Утилиты" в инспекторе атрибутов измените значение стиля
  • (необязательно) Изменение других значений, таких как Выбор и Аксессуар

Новый iOS 6.0 dequeueReusableCellWithIdentifier:forIndexPath: использует эти значения при распределении новых ячеек и их возвращении. (Протестировано на компиляции iOS 6.0 с использованием Xcode 4.5.2)

Ответ 4

Другая альтернатива, которая сохраняет один файл, - создать Nib и вместо этого использовать registerNib:forCellReuseIdentifier:.

Создание Nib очень просто: создайте новый .xib файл в Interface Builder. Удалить представление по умолчанию. Добавить объект Cell View Table. С помощью Attributes Inspector измените стиль для ячейки. (Здесь у вас также есть возможность настроить камеру дальше, настроив другие атрибуты.)

Затем в вашем представлении контроллера таблиц viewDidLoad вызовите что-то вроде:

[self.tableView registerNib:[UINib nibWithNibName:@"StyleSubtitleTableCell" bundle:[NSBundle mainBundle]] forCellReuseIdentifier:@"Cell"];

Ответ 5

Болотный ответ правильный. Простой и вам не нужно создавать XIB файл.

Я просто хотел обновить его ответ для тех, кто делает это, используя Swift вместо Objective-C:

override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
    super.init(style: .value1, reuseIdentifier: reuseIdentifier)
}

required init?(coder aDecoder: NSCoder) {
    fatalError("init(coder:) has not been implemented")
}

Ответ 6

Моим решением является вызов initWithStyle: reuseIdentifier: после того, как я его получил, используя [self.tableView dequeueReusableCellWithIdentifier:@"cellId" forIndexPath:indexPath]. В конце концов, init - это еще один селектор, и компилятор не делает никаких ограничений на его вызов на уже инициализированном объекте. Однако он будет жаловаться на то, что не будет использовать результат вызова init, поэтому я:

UITableViewCell* cell = [self.tableView dequeueReusableCellWithIdentifier:@"cellId" forIndexPath:indexPath];
cell = [cell initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:@"cellId"];

Я предполагаю, что это не сработает в Swift...