Как отправить данные из UITableViewCell в ViewController в Swift 5

#uitableview #swift5

#uitableview #swift5

Вопрос:

У меня есть один UISlider и один UILabel. Я хочу отразить значение из этого пользовательского интерфейса в моей UILabel. У меня есть UITableViewCell, где у меня есть оба выхода, и у меня есть ViewController, к которому я подключил ползунок, и я создаю функцию действия (valueChanged). Я выполнил логику в этой функции, но не работает, потому что говорит, что мой выход равен НУЛЮ. Кто-нибудь может мне помочь правильно передать эти данные? Возможно, это можно сделать с помощью делегатов, но я не знаю, как точнее.

Вот мой код для UITableViewCell:

 import UIKit

class SliderCell: UITableViewCell {

    // Interface Links
    @IBOutlet weak var slider: UISlider!
    @IBOutlet weak var progressLabel: UILabel!

    override func awakeFromNib() {
        super.awakeFromNib()
    }

    override func setSelected(_ selected: Bool, animated: Bool) {
        super.setSelected(selected, animated: animated)
    }
}
  

И вот контроллер:

 import UIKit

// Table View with Dynamic Cells
class MainViewController: UITableViewController {

    var sliderCell: SliderCell!

    override func viewDidLoad() {
        super.viewDidLoad()

        sliderCell = SliderCell()
    }

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

        switch indexPath.row {
        // Row 0  -  My Slider
        case 0:
            let cell = tableView.dequeueReusableCell(withIdentifier: "sliderCell", for: indexPath) as! SliderCell
            return cell
        default:
            return UITableViewCell()
        }
    }

    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return 1
    }

    override func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {

        // Set the height for slider cell to 200 pixels
        switch indexPath.row {
        case 0:
            return 200
        default:
            return 200
        }
    }

    // Reflect the value of Slider to ProgressLabel
    @IBAction func sliderValueChanged(_ sender: UISlider) {

        var currentValue = Int(sender.value)
        let stepSize:Int = 10
        currentValue = (currentValue - currentValue % stepSize)
        DispatchQueue.main.async {
            self.sliderCell.progressLabel.text = "(currentValue)%"
        }
    }
}
  

Заранее спасибо, если вы это читаете!

Комментарии:

1. Если табличное представление будет содержать только одну ячейку, почему вы не используете статическую ячейку с выходами (или почему вы вообще используете табличное представление? 😉)

2. Потому что в моем проекте есть 20 ячеек, которые необходимо скрыть в разных ситуациях. Это всего лишь простой пример того, что мне нужно.

Ответ №1:

Не добавляйте здесь ссылку на свою ячейку

 sliderCell = SliderCell()
  

но используйте делегат в cellForRowAt, например

   let cell = 
  cell.delegate = self
  

затем внутри ячейки отправьте что-нибудь с

 delegate?.sendSliderValue()
  

вы также можете добавить реализацию действия слайдера внутри vc, например

 let cell = //
cell.slider.addTarget ////
cell.slider.tag = indexPath.row

 @objc func sliderChanged(_ slider:UISlider) { 
     print(slider.value,slider.tag)
     if let cell = tableView.cellForRow(at:IndexPath(row:slider.tag,section:0)) as? SliderCell {
         var currentValue = Int(sender.value)
         let stepSize:Int = 10
         currentValue = (currentValue - currentValue % stepSize)
         cell.progressLabel.text = "(currentValue)"
     }

     // or 

     if let cell = tableView.visibleCells.first as? SliderCell {
       //////
     }
 }
  

Комментарии:

1. Второй пример с addTarget работает для меня. В примере с делегатом я не смог это реализовать. Теперь, как я могу отразить значение ползунка на UILabel, который находится в UITableViewCell? Спасибо Sh_Khan!

2. Мне удалось сделать это с помощью делегирования и протокола! Я так счастлив! : D Спасибо!