6

我想对 UIViewController 内的 UICollectionView 进行单元测试(例如,我想测试 UICollectionView 是否具有我期望它在单元测试中具有的单元格数量)

我的单元测试基于以下博客(关于如何对视图控制器进行单元测试):http: //yetanotherdevelopersblog.blogspot.co.il/2012/03/how-to-test-storyboard-ios-view.html

在单元测试中,我能够在我正在测试的 UIViewController 中获得指向 UICollectionView 的指针。这是我测试的 setUp 方法中的代码:

storyboard = [UIStoryboard storyboardWithName:@"MainStoryboard" bundle:nil];
viewController = [self.storyboard instantiateViewControllerWithIdentifier:@"MainViewController"];
[viewController performSelectorOnMainThread:@selector(loadView) withObject:nil waitUntilDone:YES];
self.collectionView = viewController.collectionView

不幸的是,此 UICollectionView 中没有单元格(即self.collectionView.visibleCells.count等于 0,因此我无法访问单元格并测试数据是否符合我的预期),尽管在调试时我可以看到我的应用程序将这些单元格添加到集合中看法。

我究竟做错了什么?关于如何在 UIViewController 中对 UICollectionView 进行单元测试的任何提示和技巧?

编辑: 在 Roshan 的帮助下,我现在可以缩小我的问题范围。我正在对具有 UICollectionView 的 UIViewController 进行单元测试。在我的单元测试中,我想测试 CollectionView 中单元格的值是否符合我的预期。但是,collectionView:cellForItemAtIndexPath:没有在我的 ViewController 上调用,因此当它作为单元测试调用时我的单元格不存在。任何想法?

4

2 回答 2

4

在 iOS 上实现 TTD 时,尽量不要依赖系统调用委托和数据源方法。

您应该直接从单元测试中调用这些方法。只需为您的测试配置正确的环境即可。

例如,当我为 UICollectionView 实现 TDD 时,我创建了两个单独的类,专门用于实现 UICollectionViewDataSource 和 UICollectionViewDelegate 协议,创建关注点分离,我可以将这些类分别单元测试到视图控制器本身,尽管我仍然需要初始化视图控制器来设置视图层次结构。

这是一个示例,当然省略了标题和其他次要代码。

UICollectionViewDataSource 示例

@implementation CellContentDataSource

@synthesize someModelObjectReference = __someModelObjectReference;

#pragma mark - UICollectionViewDataSource Protocol implementation

- (NSInteger) collectionView:(UICollectionView *)collectionView
  numberOfItemsInSection:(NSInteger)section
{
    return __someModelObjectReference ? 
         [[__someModelObjectReference modelObjects] count] : 0;
}

- (UICollectionViewCell *) collectionView:(UICollectionView *)collectionView
               cellForItemAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString * const kCellReuseIdentifer =   @"Cell";
    UICollectionViewCell *cell = [collectionView dequeueReusableCellWithReuseIdentifier:kCellReuseIdentifer forIndexPath:indexPath];

    ModelObject *modelObject = [__someModelObjectReference modelObjects][[indexPath item]];

    /* Various setter methods on your cell with the model object */

    return cell;
}
@end

单元测试示例

- (void) testUICollectionViewDataSource
{
    UIStoryBoard *storyboard = [UIStoryboard storyboardWithName:@"MainStoryboard" bundle:nil];
    MainViewController *viewController = [storyboard instantiateViewControllerWithIdentifier:@"MainViewController"];
    [viewController view]; // Loads the view hierarchy

    // Using OCMock to mock the returning of the model objects from the model object reference.
    // The helper method referenced builds an array of test objects for the collection view to reference
    id modelObjectMock = [OCMockObject mockForClass:[SomeModelObjectReference class]];
    [[[modelObjectMock stub] andReturn:[self buildModelObjectsForTest]] modelObjects];

    CellContentDataSource *dataSource = [CellContentDataSource new];
    dataSource.someModelObjectReference = modelObjectMock;
    viewController.collectionView.dataSource = dataSource;

    // Here we call the data source method directly
    UICollectionViewCell *cell = [dataSource collectionView:viewController.collectionView cellForItemAtIndexPath:[NSIndexPath indexPathForItem:0 inSection:0]];

    XCTAssertNotNil(cell, @"Cell should not be nil");
    // Assert your cells contents here based on your test model objects
}
于 2014-02-12T18:18:08.333 回答
3

我不确切知道问题出在哪里,但 Apple 的文档说明了这一点loadView

你不应该直接调用这个方法。视图控制器在请求视图属性但当前为 nil 时调用此方法。如果您手动创建视图,则必须覆盖此方法并使用它来创建视图。如果你使用 Interface Builder 创建你的视图并初始化视图控制器——也就是说,你使用 initWithNibName:bundle: 方法初始化视图,直接设置 nibName 和 nibBundle 属性,或者在 Interface Builder 中创建你的视图和视图控制器——那么你不能重写这个方法。

因此,不要直接调用它,而是调用[viewController view]强制加载视图。

至于您的问题,请检查是否viewController.collectionView为0。您的插座可能尚未设置。

于 2013-10-27T15:56:58.430 回答