单元测试自定义 UITableViewCell?

unit test custom UITableViewCell?

是否可以只初始化和加载自定义单元和测试插座?

我的 ViewControllerTableView 和分开的 dataSource (这是自定义数据源的子类)。所以使用所有这些创建单元格有点棘手。

自定义单元格只有几个标签和用于从对象更新它们的配置方法,因此如果加载,测试会很容易。

针对它执行单元测试并不值得麻烦。但是,有一种更简单的方法可以解决这个问题。

您可以创建一个视图模型来支持您的单元格,然后测试该视图模型是否为每个项目提供了正确的值。

此处是填充两个标签和图像的视图模型的简单示例:

class MyCellModel {
    var stringOne: String? {
        return "Compute string 1"
    }

    var stringTwo: String? {
        return "Compute string 2"
    }

    var image: UIImage? {
        return UIImage(named: "myimage")
    }
}

使用此模型,您可以将用于生成这些值的逻辑放在相关的计算属性中。然后出于测试目的,您可以使用要测试的值初始化此模型。

可以为自定义 UITableViewCell 编写单元测试,以测试其插座和其中包含的任何其他功能。以下示例对此进行了演示:

class TestItemTableViewCell: XCTestCase {

var tableView: UITableView!
private var dataSource: TableViewDataSource!
private var delegate: TableViewDelegate!

override func setUp() {
    tableView = UITableView(frame: CGRect(x: 0, y: 0, width: 200, height: 400), style: .plain)
    let itemXib = UINib.init(nibName: "ItemTableViewCell",
                                    bundle: nil)
    tableView.register(itemXib,
                       forCellReuseIdentifier: "itemCell")
    dataSource = TableViewDataSource()
    delegate = TableViewDelegate()
    tableView.delegate = delegate
    tableView.dataSource = dataSource
}

func testAwakeFromNib() {
    let indexPath = IndexPath(row: 0, section: 0)
    let itemCell = createCell(indexPath: indexPath)

    // Write assertions for things you expect to happen in
    // awakeFromNib() method.
}

}


extension TestItemTableViewCell {

func createCell(indexPath: IndexPath) -> ItemTableViewCell {
    
    let cell = dataSource.tableView(tableView, cellForRowAt: indexPath) as! ItemTableViewCell
    XCTAssertNotNil(cell)
    
    let view = cell.contentView
    XCTAssertNotNil(view)
    
    return cell
}
}

private class TableViewDataSource: NSObject, UITableViewDataSource {

var items = [Item]()

override init() {
    super.init()
    
    // Initialize model, i.e. create&add object in items.
}

func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    return items.count
}

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    let cell = tableView.dequeueReusableCell(withIdentifier: "itemCell",
                                             for: indexPath)
    return cell
}
}

private class TableViewDelegate: NSObject, UITableViewDelegate {

 }

此方法模仿 UITableViewCell 在运行时 created/reused 的方式。调用相同的方法,例如awakeFromNibIBOutlets 已初始化等。我相信您甚至可以测试单元格的大小(例如 height),即使我还没有尝试过。请注意,拥有一个包含模型对象“可视化”逻辑的视图模型是一种很好的模块化方法,并且可以更轻松地对部分代码进行单元测试(如上面另一个答案中所述)。但是,对于视图模型对象的单元测试,您无法测试 UITableViewCell.

的整个生命周期