5

我在 UICollectionView 的 batchUpdate 操作中发现了一个简单的边缘案例,它应该可以工作但失败了

尝试执行插入和移动到相同的索引路径({length = 2, path = 0 - 2})

我的操作是从 [A, B] --> [C, B', A] 走。这是通过更新完成的:

  • 从索引 0 移动到 2
  • 重新加载索引 1
  • 索引 0 处的插入

显然错误是错误的,插入索引与移动 TO 索引不同。

我设置了演示以确保这是一个 UICollectionView 问题,如果您希望看到它的实际效果,这是我的代码:

@implementation ViewController {
  UICollectionView *_collection;
  NSArray *_values;
}

- (instancetype)init
{
  self = [super init];
  if (self) {
    UICollectionViewFlowLayout *layout = [[UICollectionViewFlowLayout alloc] init];
    _collection = [[UICollectionView alloc] initWithFrame:CGRectZero collectionViewLayout:layout];
    _collection.dataSource = self;
    _values = @[@1, @2];
    [_collection registerClass:[UICollectionViewCell class]
      forCellWithReuseIdentifier:@"reuse"];
    [self.view addSubview:_collection];
  }
  return self;
}

- (void)viewDidLoad {
  [super viewDidLoad];
  _collection.frame = self.view.bounds;

  double delayInSeconds = 2.0;
  dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC));
  dispatch_after(popTime, dispatch_get_main_queue(), ^(void){
    [self triggerBatchUpdate];
  });
}

- (void)triggerBatchUpdate {
  [_collection performBatchUpdates:^{
    _values = @[@4, @5, @1];
    [_collection insertItemsAtIndexPaths:@[[self index:0]]];
    [_collection moveItemAtIndexPath:[self index:0] toIndexPath:[self index:2]];

    // Works with this line
//    [_collection moveItemAtIndexPath:[self index:1] toIndexPath:[self index:1]];

    // Fails with this line
    [_collection reloadItemsAtIndexPaths:@[[self index:1]]];
  } completion:nil];
}

- (NSIndexPath *)index:(NSUInteger)ind {
  return [NSIndexPath indexPathForRow:ind inSection:0];
}

- (NSInteger)collectionView:(UICollectionView *)collectionView
     numberOfItemsInSection:(NSInteger)section {
  return _values.count;
}

- (UICollectionViewCell *)collectionView:(UICollectionView *)collectionView
                  cellForItemAtIndexPath:(NSIndexPath *)indexPath {
  UICollectionViewCell *cell = [_collection dequeueReusableCellWithReuseIdentifier:@"reuse"
                                                                      forIndexPath:indexPath];
  cell.backgroundColor = [UIColor grayColor];
  return cell;
}


@end

如果我使用代码有效的事实

[_collection moveItemAtIndexPath:[self index:1] toIndexPath:[self index:1]];

代替

[_collection reloadItemsAtIndexPaths:@[[self index:1]]];

让我怀疑。这两个操作应该是等价的。

知道这里发生了什么吗?我相信这是一个 UICollectionView 错误吗?

编辑:事实证明与此有关的另一个崩溃:

尝试从同一索引路径({length = 2, path = 0 - 5})执行删除和移动

4

1 回答 1

8

这似乎是一个错误,尽管文档performBatchUpdates:completion解释了插入和删除操作中索引的上下文,并提到允许重新加载操作,但它没有详细说明重新加载操作中索引的上下文。

经过一些实验,似乎“在幕后”重新加载被实现为删除和插入。这似乎会导致一些其他操作与重新加载索引之间存在重叠的问题。

但是,我确实发现用显式删除和插入替换重新加载似乎有效,因此您可以使用:

- (void)triggerBatchUpdate {
    [_collection performBatchUpdates:^{
        _values = @[[UIColor blueColor], [UIColor yellowColor], [UIColor redColor]];
        [_collection moveItemAtIndexPath:[self index:0] toIndexPath:[self index:2]];
        [_collection insertItemsAtIndexPaths:@[[self index:0]]];

        [_collection deleteItemsAtIndexPaths:@[[self index:1]]];
        [_collection insertItemsAtIndexPaths:@[[self index:1]]];
    } completion:nil];
}
于 2017-02-10T01:21:48.513 回答