将数据从TableView发送到DetailView Swift

时间:2015-02-10 11:35:11

标签: swift tableview detailview

到目前为止,我正试图为我做一件最简单,最令人困惑的事情 我想开发自己的应用程序,为了做到这一点,我需要能够传递一些信息,具体取决于用户点击哪一行(它是Swift语言)

我们有一个RootViewController(表视图)和一个DetailViewController(带有1个标签和1个图像) enter image description here

(我们的观点)

以下是代码:

@IBOutlet weak var tableView: UITableView!


var vehicleData : [String] = ["Ferrari 458" , "Lamborghini Murcielago" , "Bugatti Veyron", "Mercedes Benz Biome"]

override func viewDidLoad() {
    super.viewDidLoad()
    // Do any additional setup after loading the view, typically from a nib.

    var nib = UINib(nibName: "TableViewCell", bundle: nil)

    tableView.registerNib(nib, forCellReuseIdentifier: "cell")



}

func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    return vehicleData.count
}



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


    let cell:TableViewCell = self.tableView.dequeueReusableCellWithIdentifier("cell") as TableViewCell

    cell.lblCarName.text = vehicleData[indexPath.row]

    cell.imgCar.image = UIImage(named: vehicleData[indexPath.row])

    return cell   
}

func tableView(tableView: UITableView, didSelectRowAtIndexPath indexPath: NSIndexPath) {

    performSegueWithIdentifier("DetailView", sender: self)
}

 override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) {

    if(segue.identifier == "DetailView") {

        var vc = segue.destinationViewController as DetailViewController

    }

}

func tableView(tableView: UITableView, heightForRowAtIndexPath indexPath: NSIndexPath) -> CGFloat {
    return 100
}

自定义TableViewCell类(具有带单元格的xib文件)

class TableViewCell: UITableViewCell {


@IBOutlet weak var lblCarName: UILabel!

@IBOutlet weak var imgCar: UIImageView!

override func awakeFromNib() {
    super.awakeFromNib()
    // Initialization code
}

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

    // Configure the view for the selected state
}

class DetailViewController: UIViewController {



    @IBOutlet weak var lblDetail: UILabel!

    @IBOutlet weak var imgDetail: UIImageView!

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

问题是:

如果用户点击法拉利458,DetailViewController中的lblDetail会显示:法拉利458是一辆超级跑车,能够达到325公里/小时......(无论我们想要什么)  和imgDetail将能够显示汽车的图像(无论我们想要什么)

如果用户现在点击Bugatti Veyron,那么lblDetail会向我们展示:Bugatti Veyron是一款完美的超级运动机器。它是世界上最快的汽车之一....

imgDetail向我们展示了这辆车的图片

所有汽车都取决于我们点击了哪一行

我知道第一个View Controller中的prepareForSegue func工作正在进行,但我尝试了很多不同的方法使它成为可能而且运行正常

我们怎么做?

4 个答案:

答案 0 :(得分:39)

以下是您的示例:

var valueToPass:String!

func tableView(tableView: UITableView!, didSelectRowAtIndexPath indexPath: NSIndexPath!) {
    println("You selected cell #\(indexPath.row)!")

    // Get Cell Label
    let indexPath = tableView.indexPathForSelectedRow!
    let currentCell = tableView.cellForRowAtIndexPath(indexPath)! as UITableViewCell

    valueToPass = currentCell.textLabel.text
    performSegueWithIdentifier("yourSegueIdentifer", sender: self)
}

override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?){

    if (segue.identifier == "yourSegueIdentifer") {
        // initialize new view controller and cast it as your view controller
        var viewController = segue.destinationViewController as AnotherViewController
        // your new view controller should have property that will store passed value
        viewController.passedValue = valueToPass
    }
}

但请不要忘记在passedValue中创建DetailViewController变量。

这只是将数据从一个viewController传递到另一个viewController的示例,您可以根据需要使用此示例传递数据。

有关更多信息,请参阅此链接。

Passing values between ViewControllers based on list selection in Swift

Use didSelectRowAtIndexPath or prepareForSegue method for UITableView?

Swift: Pass UITableViewCell label to new ViewController

https://teamtreehouse.com/forum/help-swift-segue-with-variables-is-not-working

可能会对你有帮助。

Swift 3.0

var valueToPass:String!
func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
    print("You selected cell #\(indexPath.row)!")

    // Get Cell Label
    let indexPath = tableView.indexPathForSelectedRow!
    let currentCell = tableView.cellForRow(at: indexPath)! as UITableViewCell

    valueToPass = currentCell.textLabel?.text
    performSegue(withIdentifier: "yourSegueIdentifer", sender: self)
}

func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?){

    if (segue.identifier == "yourSegueIdentifer") {
        // initialize new view controller and cast it as your view controller
        var viewController = segue.destination as! AnotherViewController
        // your new view controller should have property that will store passed value
        viewController.passedValue = valueToPass
    }
}

答案 1 :(得分:3)

这可能是另一种解决方案,在didSelectRowAtIndexPath方法中没有太多代码。 请注意,虽然它看起来更干净,并且我们不需要额外的变量valueToPass,但这可能不是最佳实践,因为performSegue方法内部的sender参数应该是发起segue的实际对象(或nil)。 / p>

// MARK: UITableViewDelegate methods

func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
    tableView.deselectRow(at: indexPath, animated: true)

    performSegue(withIdentifier: "goToSecondVC", sender: indexPath)
}

// MARK: UIViewController methods

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    if segue.identifier == "goToSecondVC" {    
        if segue.destination.isKind(of: CarDetailsController.self) {
            let secondVC = segue.destination as! CarDetailsController

            let indexPath = sender as! IndexPath

            secondVC.passedValue = carsArray[indexPath.row]
        }
    }
}

答案 2 :(得分:1)

如果将segue从原型单元格(在Interface Builder中)拖动到下一个View Controller并将其segue标识符设置为"您的Segue Identifier",您也可以使用此快捷方式执行此操作: / p>

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    if segue.identifier == "Your Segue Identifier" {
        let cell = sender as! YourCustomCell
        let vc = segue.destination as! PushedViewController
        vc.valueToPass = cell.textLabel?.text // or custom label
    }
}

您也不需要performSegueWithIdentifier()中的didSelectRowAtIndexPath(),也不需要此表格查看方法。

PushedViewController.swift(下一个视图控制器)中:

var valueToPass: String!

override func viewDidLoad() {
    super.viewDidLoad()
    yourLabel.text = valueToPass
}

在从故事板初始化之后设置标签的值非常重要。这意味着,您无法直接在之前的View Controller prepareForSegue()中设置标签,因此需要将其与valueToPass一起传递。

答案 3 :(得分:0)

很简单,我在上面的回答中加了一句话。 要在详细视图标签中获取所选车辆名称,

lblDetail.text = passedValue

您可以在详细视图的viewDidLoad()函数中添加此行代码。 passedValue包含用户选择的汽车名称(在prepareForSegue中指定),然后您可以分配给您的详细视图标签。

希望它有所帮助!!