单击表视图单元格上的按钮时,如何获取indexPath.row?

时间:2019-05-03 15:15:21

标签: ios swift uitableview

This is my app

您好,上面的图像是我的待办事项列表应用程序的用户界面,现在,我只想在单击tableviewcell中的详细信息按钮时显示项目的详细信息(第一个项目,第二个项目等)。因此,为了获得项目的属性,我需要知道我刚刚单击明细按钮的行的indexPath

我已经尝试了诸如didSelectRowAtindexPathForSelectedRow之类的表视图属性,但是两者都不起作用。对于didSelectRowAt用户,需要首先单击该行,然后单击“详细信息”按钮,这不是我想要的,而indexPathForSelectedRow对我不起作用。

2 个答案:

答案 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保存在单元格本身(或更糟的是,“标记”值)中。这样,可以防止您在以后插入表或从表中删除行时使其不对齐。


顺便说一句,我使用了相当通用的方法/闭包名称。在真实的应用中,您可以给它们起更有意义的名称,例如didTapInfoButtondidTapSaveButton等),以阐明功能意图。

答案 1 :(得分:1)

实施委托方法tableView(_:accessoryButtonTappedForRowWith:)

func tableView(_ tableView: UITableView, accessoryButtonTappedForRowWith indexPath: IndexPath)

但是,如果您要导航到其他控制器,则将segue连接到附件查看按钮

如果该按钮是自定义按钮,请在Issue Detecting Button cellForRowAt 中查看我的答案