Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

unit test cellForRowAtIndexPath when using storyBoards

If I'm dequeuing a cell from an identifier in a storyboard, how in a unit testing way can I call cellForRowAtIndexPath and not have the cell be nil?

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {

    MyCustomTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:kCustomCell];

    cell.guestNameText.text = self.details.guestName;

    return cell;
}

Not working, put a break point above after dequeReusableCell is called and cell is nil:

ETA: UPDATED WORKING CODE TO PASS THE TEST:

- (void)setUp {

    [super setUp];
    _detailVC_SUT = [[UIStoryboard storyboardWithName:@"MainStoryboard" bundle:nil]
     instantiateViewControllerWithIdentifier:kDetailsVC];
    _myService = [OCMockObject niceMockForClass:[MyService class]];
    _detailVC_SUT.service = _myService;
}


- (void)test_queryForDetailsSucceeded_should_set_cell_text_fields {

    [_detailVC_SUT view]; // <--- Need to load the view for this to work
    Details *details = [DetailsBuilder buildStubDetails];
    [_detailVC_SUT queryForDetailsSucceededWithDetails:details];

    [self getFirstCellForGuestName];
}

- (void)getFirstCellForGuestName {

    MyCustomTableViewCell *guestNameCell = (MyCustomTableViewCell*)[_detailVC_SUT tableView:_detailVC_SUT.detailsTableView cellForRowAtIndexPath:[NSIndexPath indexPathForRow:0 inSection:0]];

    expect(guestNameCell.guestNameText.text).to.equal(@"Mark");
}
like image 959
Mark W Avatar asked Jan 12 '23 19:01

Mark W


2 Answers

Here is how I test table views and their cells. The key here is to call beginAppearanceTransition on view controller in order to load it from the storyboard.

class MyTests: XCTestCase {
  var viewController: UIViewController!

  override func setUp() {
    super.setUp()

    let storyboard = UIStoryboard(name: "MyStoryboard", bundle: nil)
    viewController = storyboard.instantiateViewControllerWithIdentifier("myViewControllerId")
    viewController.beginAppearanceTransition(true, animated: false)
  }

  override func tearDown() {
    super.tearDown()

    viewController.endAppearanceTransition()
  }


  func testShowItemsFromNetwork() {
    //
    // Load the table view here ...
    //

    let tableView = viewController.tableView

    // Check the number of table rows

    XCTAssertEqual(3, tableView.dataSource?.tableView(tableView, numberOfRowsInSection: 0))

    // Check label text of the cell in the first row

    let indexPath = NSIndexPath(forRow: 0, inSection: 0)
    let cell = tableView.dataSource?.tableView(tableView, cellForRowAtIndexPath: indexPath)
    XCTAssertEqual("Test cell title", cell!.textLabel!.text)
  }
}
like image 184
Evgenii Avatar answered Jan 22 '23 13:01

Evgenii


The table view has a frame of CGRect.zero when it tries to call cellForRow(at:). As a result, cellForRow(at:) returns nil. In order to test it I did the following inside your test function:

func testCellForRow_AtZero_ReturnsTaskCell() {
    let mockTableView = UITableView(frame: CGRect(x: 0, y: 0, width: 320, height: 480), style: .plain)

     mockTableView.dataSource = dataSource
     mockTableView.register(YourTableViewCell.self, forCellReuseIdentifier: String(describing: YourTableViewCell.self))

     //The reload is also necessary
     mockTableView.reloadData()

     let cell = mockTableView.cellForRow(at: IndexPath(row: 0, section: 0))

     XCTAssertTrue(cell is YourTableViewCell)
}

Don't forget that in your dataSource you should have a value for the Indexpath(row: 0, section: 0). Otherwise the test would fail.

like image 34
Fernando Cardenas Avatar answered Jan 22 '23 12:01

Fernando Cardenas