您好,上面的图像是我的待办事项列表应用程序的用户界面,现在,我只想在单击tableviewcell中的详细信息按钮时显示项目的详细信息(第一个项目,第二个项目等)。因此,为了获得项目的属性,我需要知道我刚刚单击明细按钮的行的indexPath
。
我已经尝试了诸如didSelectRowAt
或indexPathForSelectedRow
之类的表视图属性,但是两者都不起作用。对于didSelectRowAt
用户,需要首先单击该行,然后单击“详细信息”按钮,这不是我想要的,而indexPathForSelectedRow
对我不起作用。
答案 0 :(得分:2)
此类问题的通用解决方案是将按钮的@IBAction
连接到单元中(而不是视图控制器中)的处理程序,然后使用委托协议模式,以便单元可以告诉表何时点击按钮。关键是当单元格执行此操作时,它将提供对自身的引用,然后视图控制器可以使用该引用来确定适当的indexPath(从而确定行)。
例如:
为您的UITableViewCell
子类提供协议:
protocol CustomCellDelegate: class {
func cell(_ cell: CustomCell, didTap button: UIButton)
}
将@IBAction
连接到单元格(而不是视图控制器),并调用该委托方法:
class CustomCell: UITableViewCell {
weak var delegate: CustomCellDelegate?
@IBOutlet weak var customLabel: UILabel!
func configure(text: String, delegate: CustomCellDelegate) {
customLabel.text = text
self.delegate = delegate
}
@IBAction func didTapButton(_ button: UIButton) {
delegate?.cell(self, didTap: button)
}
}
很显然,在创建单元格时,调用configure
方法,除其他外,将对自身的引用作为委托:
extension ViewController: UITableViewDataSource {
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { ... }
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: "CustomCell", for: indexPath) as! CustomCell
let text = ...
cell.configure(text: text, delegate: self)
return cell
}
}
最后,让委托方法调用indexPath(for:)
来确定相关单元的索引路径:
extension ViewController: CustomCellDelegate {
func cell(_ cell: CustomCell, didTap button: UIButton) {
guard let indexPath = tableView.indexPath(for: cell) else { return }
// use `indexPath.row` here
}
}
另一种方法是使用闭包,但是再次使用将按钮@IBAction
挂接到单元格的相同常规模式,但是让它调用闭包而不是委托方法:
定义带有闭包的自定义单元格,在点击按钮时将调用它:
class CustomCell: UITableViewCell {
typealias ButtonHandler = (CustomCell) -> Void
var buttonHandler: ButtonHandler?
@IBOutlet weak var customLabel: UILabel!
func configure(text: String, buttonHandler: @escaping ButtonHandler) {
customLabel.text = text
self.buttonHandler = buttonHandler
}
@IBAction func didTapButton(_ button: UIButton) {
buttonHandler?(self)
}
}
当表视图数据源创建单元格时,请提供处理程序闭包:
extension ViewController: UITableViewDataSource {
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { ... }
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: "CustomCell", for: indexPath) as! CustomCell
let text = ...
cell.configure(text: text, buttonHandler: { [weak self] cell in // the `[weak self]` is only needed if this closure references `self` somewhere
guard let indexPath = tableView.indexPath(for: cell) else { return }
// use `indexPath` here
})
return cell
}
}
我个人更喜欢委托协议模式,因为它倾向于更好地扩展,但是两种方法都有效。
请注意,在两个示例中,我都故意避免将indexPath
保存在单元格本身(或更糟的是,“标记”值)中。这样,可以防止您在以后插入表或从表中删除行时使其不对齐。
顺便说一句,我使用了相当通用的方法/闭包名称。在真实的应用中,您可以给它们起更有意义的名称,例如didTapInfoButton
,didTapSaveButton
等),以阐明功能意图。
答案 1 :(得分:1)
实施委托方法tableView(_:accessoryButtonTappedForRowWith:)
func tableView(_ tableView: UITableView, accessoryButtonTappedForRowWith indexPath: IndexPath)
但是,如果您要导航到其他控制器,则将segue连接到附件查看按钮
如果该按钮是自定义按钮,请在Issue Detecting Button cellForRowAt 中查看我的答案