14

使用 IOS 6 和新的 UICollectionView 我在 UIViewController 上有 2 个 UICollectionViews 并使用 Storyboard 当我在 UICollectionView 周围单击时出现以下错误我不确定为什么会发生这种情况,谷歌为此错误返回 0 结果...

2012-10-13 20:30:06.421 MyApp [768:907] -[UICollectionViewData numberOfItemsBeforeSection:] 中的断言失败,/SourceCache/UIKit/UIKit-2372/UICollectionViewData.m:415
2012-10-13 20:30:06.427 MyApp [768:907] 由于未捕获的异常“NSInternalInconsistencyException”而终止应用程序,原因:“当集合视图中只有 1 个部分时,请求第 2147483647 部分之前的项目数”

我根本没有使用 Sections 并将其设置为仅 1 个部分:

- (NSInteger)numberOfSectionsInCollectionView:(UICollectionView *)collectionView
{
    return 1;
}

当我单击 UICollectionView 区域时,它会发生(似乎是随机的)我有一段代码可以从 UICollectionView 中删除元素:

[tilesArray removeObjectAtIndex:indexPath.row];
[self.tilesCollectionView reloadData];        
[resultsArray addObject:str];
[self.resultsCollectionView reloadData];

这是数组如何连接到 UICollectionViews

-(NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:(NSInteger)section {

    if(collectionView.tag == 2)
        return [resultsArray count];
    else
        return [tilesArray count];
}
-(UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath {

    if(collectionView.tag == 2)
    {
        static NSString *cellIdentifier = @"ResultCell";
        ResultCell *cell = (ResultCell *)[collectionView dequeueReusableCellWithReuseIdentifier:cellIdentifier forIndexPath:indexPath];

        cell.mainImage.image = [UIImage imageNamed:currentArtContainer.tileMainImage];

        NSString *cellData = [resultsArray objectAtIndex:indexPath.row];
        [cell.titleLabel setText:cellData];

        return cell;
    }
    else
    {
        static NSString *cellIdentifier = @"TileCell";
        TileCell *cell = (TileCell *)[collectionView dequeueReusableCellWithReuseIdentifier:cellIdentifier forIndexPath:indexPath];

        cell.topImage.image = [UIImage imageNamed:currentArtContainer.tileTopImage];

        cell.mainImage.image = [UIImage imageNamed:currentArtContainer.tileMainImage];

        NSString *cellData = [tilesArray objectAtIndex:indexPath.row];
        [cell.titleLabel setText:cellData];

        return cell;
    }
}

请帮我解决这个问题。甚至不确定如何捕获异常,因为它来自 UICollectionView 内部代码。还忘了提到我正在为 UICollectionView 使用自定义布局,但我不知道它是否与这个问题有关。

再次感谢你们的帮助!

编辑:在调用所有布局方法后,我得到了异常......这是我的自定义布局......我仍然不确定如何修复它,因为我所有的 try catch 块都没有捕获错误。 ..

- (void)prepareLayout
{
    NSLog(@"In prepareLayout.");

    [super prepareLayout];
    _cellCount = [[self collectionView] numberOfItemsInSection:0];
}

- (CGSize)collectionViewContentSize
{
    NSLog(@"In collectionViewContentSize.");
    @try {

       CGSize csize = [self collectionView].frame.size;
        NSLog(@"In here.");
        return csize;
    }
    @catch(NSException *exception)
    {
        NSLog(@"collectionViewContentSize %@: %@",[exception name],[exception reason]);
    }

}

- (UICollectionViewLayoutAttributes *)layoutAttributesForItemAtIndexPath:(NSIndexPath *)indexPath
{
    @try {

    NSLog(@"In layoutAttributesForItemAtIndexPath. item %d", indexPath.row);


    UICollectionViewLayoutAttributes* attributes = [UICollectionViewLayoutAttributes layoutAttributesForCellWithIndexPath:indexPath];

    attributes.size = CGSizeMake(ITEM_SIZE, ITEM_SIZE);

    NSLog(@"firstRun %d", self.firstRun);
    //NSLog(@"panStatus %@", self.panStatus);

    if (self.firstRun == 0) {
        CGFloat x = 0, y = 0;
        CGFloat boundx = self.collectionView.frame.size.width * 0.70;   // Use the whole canvas
        CGFloat boundy = self.collectionView.frame.size.height * 0.70; // Use the whole canvas
        while (x < 50 || x > boundx) x = arc4random() % (int)boundx + indexPath.item;
        while (y < 50 || y > boundy) y = arc4random() % (int)boundy + indexPath.item;

        NSMutableDictionary *dictionary = [NSMutableDictionary dictionary];
        if (self.positions)
            dictionary = [self.positions mutableCopy];
        [dictionary setObject:[NSValue valueWithCGPoint:CGPointMake(x, y)] forKey:@(indexPath.item)];
        if (!self.positions)
            self.positions = [[NSDictionary alloc] init];
        self.positions = dictionary;

        attributes.center = CGPointMake(x, y);
    } else if (self.firstRun > 0) {
        CGPoint point = [[self.positions objectForKey:@(indexPath.item)] CGPointValue];
        attributes.center = CGPointMake(point.x, point.y);
    }

    NSLog(@"END layoutAttributesForItemAtIndexPath. item %d", indexPath.row);
        return attributes;

    }
    @catch(NSException *exception)
    {
        NSLog(@"layoutAttributesForItemAtIndexPath %@: %@",[exception name],[exception reason]);
    }
}

-(NSArray *)layoutAttributesForElementsInRect:(CGRect)rect
{
    @try {

    NSLog(@"layoutAttributesForElementsInRect ...");

    NSMutableArray* attributes = [NSMutableArray array];
    for (NSInteger i=0 ; i < self.cellCount; i++) {
        NSIndexPath* indexPath = [NSIndexPath indexPathForItem:i inSection:1];
        [attributes addObject:[self layoutAttributesForItemAtIndexPath:indexPath]];
    }
    NSLog(@"END layoutAttributesForElementsInRect ...");
        return attributes;

    }
    @catch(NSException *exception)
    {
        NSLog(@"layoutAttributesForElementsInRect %@: %@",[exception name],[exception reason]);
    }
}
4

6 回答 6

9

我在从手势识别器运行 reloadData 时发生了这种情况,在手势识别器上设置delaysTouchesBegan有帮助。

于 2013-02-03T18:39:01.530 回答
6

这里的关键是您的错误中的这个数字:

2147483647

这是 的数值NSNotFound。在代码中的某处,您正在indexOfObject:对数组执行类似操作,并将其结果作为节或项索引传递到您的集合视图或布局方法之一,这会导致错误。您问题中发布的代码中没有发生这种情况,但希望您现在能够找到它。

于 2012-10-14T07:10:13.747 回答
5

这对我有用:

dispatch_async(dispatch_get_main_queue(), ^{
    [self.collectionView reloadData];
});
于 2013-06-06T14:33:29.007 回答
4

我自己也遇到了这个断言。它触发是因为我在触摸(滑动)事件上重新加载收集数据。

最初的触摸(开始)触发了一条didHighlightItemAtIndexPath:消息,然后收集数据发生了变化(通过滑动),然后didUnhighlightItemAtIndexPath:在触摸(提升)时未能触发消息。

由于我不需要突出显示,因此解决方案是在我的控制器中实现此方法:

- (BOOL)collectionView:(UICollectionView *)collectionView shouldHighlightItemAtIndexPath:(NSIndexPath *)indexPath
{
    return NO;
}

我希望这可以帮助其他任何遇到这个问题的人。

于 2013-01-09T18:44:41.797 回答
3

永远不要在手势识别器执行的方法上重新加载 UICollectionView 的数据。遵循这个规则完全解决了这个问题。Daltsy 建议在 UICollectionView 中停用 highligting 也可以防止错误,但也会停用所有“didSelectItemAtIndexPath”委托消息。要小心,因为你可能会得到一个几乎没用的 UICollectionView - 就像我几分钟前所做的那样。

于 2013-02-09T09:58:31.460 回答
0

我没有禁用突出显示,而是发现在调用 reloadData 之前确保没有突出显示任何单元格可以防止“请求部分之前的项目数”错误。这是

UICollectionViewController.CollectionView.deselectItemAtIndexPath 方法或DeselectItem在 Xamarin 中。

我在模型数据中保存当前选中项的句柄,清除集合视图的突出显示,为这些项计算新的 NSIndexPaths,然后调用 reloadData。通过使用句柄通过其可能的新 NSIndexPath 属性找到它,再次选择先前选择的项目。

selectItemAtIndexPath:animated:scrollPosition:SelectItem在 Xamarin 中。

于 2015-12-02T09:35:01.887 回答