it-swarm.com.ru

Как определить, что анимация закончилась на UITableView beginUpdates/endUpdates?

Я вставляю/удаляю ячейку таблицы, используя insertRowsAtIndexPaths/deleteRowsAtIndexPaths, завернутый в beginUpdates/endUpdates. Я также использую beginUpdates/endUpdates при настройке rowHeight. Все эти операции анимированы по умолчанию.

Как я могу определить, что анимация закончилась при использовании beginUpdates/endUpdates?

107
pixelfreak

Как насчет этого?

[CATransaction begin];

[CATransaction setCompletionBlock:^{
    // animation has finished
}];

[tableView beginUpdates];
// do some work
[tableView endUpdates];

[CATransaction commit];

Это работает, потому что анимации tableView внутренне используют анимации CALayer. То есть они добавляют анимацию к любому открытому CATransaction. Если не существует открытого CATransaction (нормальный случай), то он запускается неявно, и заканчивается в конце текущего цикла выполнения. Но если вы начнете сами, как здесь сделано, то он будет использовать это. 

284
Rudolf Adamkovič

Swift Version


CATransaction.begin()

CATransaction.setCompletionBlock({
    do.something()
})

tableView.beginUpdates()
tableView.endUpdates()

CATransaction.commit()
27
Michael

Вы можете заключить свои операции в блок анимации UIView следующим образом:

- (void)tableView:(UITableView *)tableView performOperation:(void(^)())operation completion:(void(^)(BOOL finished))completion
{
    [UIView animateWithDuration:0.0 animations:^{

        [tableView beginUpdates];
        if (operation)
            operation();
        [tableView endUpdates];

    } completion:^(BOOL finished) {

        if (completion)
            completion(finished);
    }];
}

Кредиты для https://stackoverflow.com/a/12905114/634940 .

5
Zdenek

Возможным решением может быть наследование от UITableView, для которого вы вызываете endUpdates, и перезаписать его setContentSizeMethod, поскольку UITableView корректирует размер своего содержимого в соответствии с добавленными или удаленными строками. Этот подход также должен работать для reloadData.

Чтобы гарантировать, что уведомление отправляется только после вызова endUpdates, можно также перезаписать endUpdates и установить там флаг.

// somewhere in header
@private BOOL endUpdatesWasCalled_;

-------------------

// in implementation file

- (void)endUpdates {
    [super endUpdates];
    endUpdatesWasCalled_ = YES;
}

- (void)setContentSize:(CGSize)contentSize {
    [super setContentSize:contentSize];

    if (endUpdatesWasCalled_) {
        [self notifyEndUpdatesFinished];
        endUpdatesWasCalled_ = NO;
    }
}
4
Antoni

Еще не нашли хорошего решения (если не считать подкласс UITableView). Я решил использовать performSelector:withObject:afterDelay: на данный момент. Не идеально, но выполняет свою работу.

UPDATE: Похоже, я могу использовать scrollViewDidEndScrollingAnimation: для этой цели (это зависит от моей реализации, см. комментарий).

0
pixelfreak

Вы можете использовать tableView:willDisplayCell:forRowAtIndexPath: как:

- (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath {
    NSLog(@"tableView willDisplay Cell");
    cell.backgroundColor = [UIColor colorWithWhite:((indexPath.row % 2) ? 0.25 : 0) alpha:0.70];
}

Но это также будет вызвано, когда ячейка, которая уже находится в таблице, переместится с экрана на экран, поэтому она может быть не совсем тем, что вы ищете. Я только что просмотрел все методы делегата UITableView и UIScrollView, и, похоже, ничего не нужно обрабатывать сразу после вставки анимации в ячейку.


Почему бы просто не вызвать метод, который вы хотите вызвать, когда анимация заканчивается после endUpdates?

- (void)setDownloadedImage:(NSMutableDictionary *)d {
    NSIndexPath *indexPath = (NSIndexPath *)[d objectForKey:@"IndexPath"];
    [indexPathDelayed addObject:indexPath];
    if (!([table isDragging] || [table isDecelerating])) {
        [table beginUpdates];
        [table insertRowsAtIndexPaths:indexPathDelayed withRowAnimation:UITableViewRowAnimationFade];
        [table endUpdates];
        // --> Call Method Here <--
        loadingView.hidden = YES;
        [indexPathDelayed removeAllObjects];
    }
}
0
chown

Если вы ориентируетесь на iOS 11 и выше, вы должны использовать вместо этого UITableView.performBatchUpdates(_:completion:):

tableView.performBatchUpdates({
    // delete some cells
    // insert some cells
}, completion: { finished in
    // animation complete
})
0
Robert