单元测试自定义 UITableViewCell?
unit test custom UITableViewCell?
是否可以只初始化和加载自定义单元和测试插座?
我的 ViewController
有 TableView
和分开的 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 的方式。调用相同的方法,例如awakeFromNib
、IBOutlet
s 已初始化等。我相信您甚至可以测试单元格的大小(例如 height
),即使我还没有尝试过。请注意,拥有一个包含模型对象“可视化”逻辑的视图模型是一种很好的模块化方法,并且可以更轻松地对部分代码进行单元测试(如上面另一个答案中所述)。但是,对于视图模型对象的单元测试,您无法测试 UITableViewCell
.
的整个生命周期
是否可以只初始化和加载自定义单元和测试插座?
我的 ViewController
有 TableView
和分开的 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 的方式。调用相同的方法,例如awakeFromNib
、IBOutlet
s 已初始化等。我相信您甚至可以测试单元格的大小(例如 height
),即使我还没有尝试过。请注意,拥有一个包含模型对象“可视化”逻辑的视图模型是一种很好的模块化方法,并且可以更轻松地对部分代码进行单元测试(如上面另一个答案中所述)。但是,对于视图模型对象的单元测试,您无法测试 UITableViewCell
.