从单元格的按钮单击获取UITableViewCell的indexPath

16

请查看这张图片

我在UITableViewCell中有一个按钮(红色叉号),当点击该按钮时,我想获取UITableViewCellindexPath

目前我像这样为每个按钮分配标签:cell.closeButton.tag = indexPath.section,并且在按钮被点击时,我获取indexPath.section的值如下:

@IBAction func closeImageButtonPressed(sender: AnyObject) {
  data.removeAtIndex(sender.tag)
  tableView.reloadData()
}

这是正确的实现方式吗?还有其他更优雅的方法吗?


顺便问一下,为什么你在删除一个部分后要重新加载整个表格?只需要调用tableView.deleteSections(NSIndexSet(index: sender.tag), withRowAnimation: .None)即可。 - Ozgur Vatansever
是的,我可以做到。 - Anirudha Mahale
也许你可以看一下以下的线程:http://stackoverflow.com/a/41629633/387507 - valvoline
你能解决这个问题吗? - Bista
@Mr.Bista 是的,我做了。 - Anirudha Mahale
10个回答

55

使用委托:

MyCell.swift:

import UIKit

//1. delegate method
protocol MyCellDelegate: AnyObject {
    func btnCloseTapped(cell: MyCell)
}

class MyCell: UICollectionViewCell {
    @IBOutlet var btnClose: UIButton!

    //2. create delegate variable
    weak var delegate: MyCellDelegate?

    //3. assign this action to close button
    @IBAction func btnCloseTapped(sender: AnyObject) {
        //4. call delegate method
        //check delegate is not nil with `?`
        delegate?.btnCloseTapped(cell: self)
    }
}

MyViewController.swift:

//5. Conform to delegate method
class MyViewController: UIViewController, MyCellDelegate, UITableViewDataSource,UITableViewDelegate {

    //6. Implement Delegate Method
    func btnCloseTapped(cell: MyCell) {
        //Get the indexpath of cell where button was tapped
        let indexPath = self.collectionView.indexPathForCell(cell)
        print(indexPath!.row)
    }

    func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {

        let cell = tableView.dequeueReusableCellWithIdentifier("MyCell") as! MyCell
        //7. delegate view controller instance to the cell
        cell.delegate = self

        return cell
    }
}

6

如何在Swift 4中使用按钮选择器获取点击按钮的单元格indexPath

@objc func buttonClicked(_sender:UIButton){

 let buttonPosition = sender.convert(CGPoint.zero, to: self.tableView)
 let indexPath = self.tableView.indexPathForRow(at:buttonPosition)
 let cell = self.tableView.cellForRow(at: indexPath) as! UITableViewCell
 print(cell.itemLabel.text)//print or get item
}

4

尝试最佳使用Swift闭包:简单、快速和易于理解。

在cellForRowAtIndexPath方法中:

let cell = tableView.dequeueReusableCell(withIdentifier: "CustomCellIdentifier", for: indexPath) as! CustomCell

cell.btnTick.mk_addTapHandler { (btn) in
     print("You can use here also directly : \(indexPath.row)")
     self.btnTapped(btn: btn, indexPath: indexPath)
}

在cellForRowAtIndexPath方法之外使用的选择器方法:

func btnTapped(btn:UIButton, indexPath:IndexPath) {
    print("IndexPath : \(indexPath.row)")
}

UIButton的扩展:

extension UIButton {

    private class Action {
        var action: (UIButton) -> Void

        init(action: @escaping (UIButton) -> Void) {
            self.action = action
        }
    }

    private struct AssociatedKeys {
        static var ActionTapped = "actionTapped"
    }

    private var tapAction: Action? {
        set { objc_setAssociatedObject(self, &AssociatedKeys.ActionTapped, newValue, .OBJC_ASSOCIATION_RETAIN) }
        get { return objc_getAssociatedObject(self, &AssociatedKeys.ActionTapped) as? Action }
    }


    @objc dynamic private func handleAction(_ recognizer: UIButton) {
        tapAction?.action(recognizer)
    }


    func mk_addTapHandler(action: @escaping (UIButton) -> Void) {
        self.addTarget(self, action: #selector(handleAction(_:)), for: .touchUpInside)
        tapAction = Action(action: action)

    }
}

3
在Swift 4中,只需使用以下代码:
func buttonTapped(_ sender: UIButton) {

    let buttonPostion = sender.convert(sender.bounds.origin, to: tableView)

    if let indexPath = tableView.indexPathForRow(at: buttonPostion) {
        let rowIndex =  indexPath.row
    }
}

0
//
//  ViewController.swift
//  Table
//
//  Created by Ngugi Nduung'u on 24/08/2017.
//  Copyright © 2017 Ngugi Ndung'u. All rights reserved.
//

import UIKit

class ViewController: UITableViewController{

    let identifier = "cellId"
    var items = ["item1", "2", "3"]
    override func viewDidLoad() {
        super.viewDidLoad()
        // Do any additional setup after loading the view, typically from a nib.
        self.title = "Table"
        tableView.register(MyClass.self, forCellReuseIdentifier: "cellId")
    }

    //Return number of cells you need
    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int{
        return items.count
    }

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "cellId", for: indexPath) as! MyClass
        cell.controller = self
        cell.label.text  = items[indexPath.row]
        return cell

    }

    // Delete a cell when delete button on cell is clicked 
    func delete(cell: UITableViewCell){
        print("delete")
        if let deletePath = tableView.indexPath(for: cell){
            items.remove(at: deletePath.row)
            tableView.deleteRows(at: [deletePath], with: .automatic)
        }

    }

}

class MyClass : UITableViewCell{
    var controller : ViewController?
    override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
            super.init(style: style, reuseIdentifier: reuseIdentifier)
            setUpViews()
    }

    required init?(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
        fatalError("init(coder:) has not been implemented")
    }

    let label : UILabel = {
        let label = UILabel()
        label.text = "My very first cell"
        label.translatesAutoresizingMaskIntoConstraints = false
        return label
    }()

    let btn : UIButton = {
       let bt = UIButton(type: .system)
        bt.translatesAutoresizingMaskIntoConstraints = false
        bt.setTitle("Delete", for: .normal)


        bt.setTitleColor(.red, for: .normal)
        return bt
    }()

    func handleDelete(){
       controller?.delete(cell: self)
    }
    func setUpViews(){
        addSubview(label)
        addSubview(btn)
        btn.addTarget(self, action: #selector(MyClass.handleDelete), for: .touchUpInside)
        btn.rightAnchor.constraint(equalTo: self.rightAnchor).isActive = true
        label.leftAnchor.constraint(equalTo: self.leftAnchor, constant: 16).isActive = true
        label.widthAnchor.constraint(equalTo: self.widthAnchor , multiplier: 0.8).isActive = true
        label.rightAnchor.constraint(equalTo: btn.leftAnchor).isActive = true

    }
}

这里有一个完整的示例,将回答你的问题。


0

试试这个:

override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    var cell = (tableView.dequeueReusableCell(withIdentifier: "MainViewCell", forIndexPath: indexPath) as! MainTableViewCell)
    cell.myButton().addTarget(self, action: Selector("myClickEvent:event:"), forControlEvents: .touchUpInside)
    return cell
}

这个函数用于获取行点击的位置
@IBAction func myClickEvent(_ sender: Any, event: Any) {
    var touches = event.allTouches()!
    var touch = touches.first!
    var currentTouchPosition = touch.location(inView: feedsList)
    var indexPath = feedsList.indexPathForRow(atPoint: currentTouchPosition)!
    print("position:\(indexPath.row)")
}

0

你也可以通过以下方式从 CGPoint 获取 NSIndexPath

@IBAction func closeImageButtonPressed(sender: AnyObject) {
    var buttonPosition = sender.convertPoint(CGPointZero, to: self.tableView)
    var indexPath = self.tableView.indexPathForRow(atPoint: buttonPosition)!
}

CGPointZero是一个位置常量,其坐标为(0,0)。零点等同于CGPointMake(0,0)。它将把你的buttonPosition转换成indexpath - Bhavin Ramani

0
创建一个自定义的 UIButton 类,并声明一个类似这样的存储属性,使用它来检索从 callFroRowAtIndexPath 分配的 indexPath。
class VUIButton: UIButton {
    var indexPath: NSIndexPath = NSIndexPath()
}

这是一种完整的证明解决方案,无论什么情况下都不会出现indexPath错误。试试看。


0

在你的cellForRow方法中:

#import <objc/runtime.h>

func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
    setAssociatedObject(object: YOURBUTTON, key: KEYSTRING, value: indexPath)
}

@IBAction func closeImageButtonPressed(sender: AnyObject) {
    let val = getAssociatedObject(object: sender, key: KEYSTROKING)
}

这里的val是您的indexPath对象,您可以传递任何对象,例如您可以分配传递cell对象并在按钮操作中获取它。


0
class MyCell: UICollectionViewCell {

   @IBOutlet weak var btnPlus: UIButton!

}

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> 
 UITableViewCell {


cell.btnPlus.addTarget(self, action: #selector(increment_Action(sender:)), 
 for: .touchUpInside)
cell.btnPlus.tag = indexPath.row
cell.btnPlus.superview?.tag = indexPath.section
}

@objc func increment_Action(sender: UIButton) {
        
let btn = sender as! UIButton
let section = btn.superview?.tag ?? 0

let row = sender.tag
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接