uitableviewcell笔尖中的依赖注入

时间:2017-01-01 08:17:51

标签: ios swift uitableview dependency-injection swift3

我正在尝试构建一个非常复杂的表。该表的一个单元格导入另一个表。在这个导入的表中,我根据情况显示不同的行。导入的表和所有可导入的表格单元格被组织成自己的nib文件,每个单元格都有自己的控制器。作为一名优秀的程序员,我正在尝试在整个项目中使用依赖注入。现在的问题是当我使用通常的方式在viewDidLoad()中注册nib时

let cellNib = UINib(nibName: "BatchConsumptionCell", bundle: nil)
tableView.register(cellNib, forCellReuseIdentifier: "BatchConsumptionCell")

然后在tableView(tableView:cellForRowAt :)中将它们用作dequeueReusableCell(withIdentifier:for :)

let batchConsumptionCell = tableView.dequeueReusableCell(withIdentifier: "BatchConsumptionCell", for: indexPath) as! BatchConsumptionCell
batchConsumptionCell.setConsumption(consumption: consumption)
return batchConsumptionCell

我无法及时注入依赖项。

在BatchConsumptionCell中,我在tableView(tableView:cellForRowAt :)中所做的一切正常。执行dequeueReusableCell(withIdentifier:for :)后调用此函数。但是,一旦我试图使tableView(tableView:numberOfRowsInSection :)动态我遇到问题。这个函数似乎在dequeueReusableCell(withIdentifier:for :)之前被调用,因此此时不会注入依赖项。

我试图在我的BatchConsumptionCell中覆盖init(nibName:bundle :)初始化程序,但这是一个UITableViewCell,所以我无法覆盖它。

我该如何处理?当nib及其控制器被初始化时,有没有什么方法可以注入依赖项?或者我组织我的细胞都错了?任何想法都会非常感激。

为了更清楚,这里是我的代码:

ConsumptionDetailViewController

import UIKit

class ConsumptionDetailViewController: UITableViewController {

// MARK: - Properties

var moc: NSManagedObjectContext!

var consumption: Consumption!


// MARK: - Outlet Properties

@IBOutlet weak var labelDate: UILabel!
@IBOutlet weak var labelTime: UILabel!
...


// MARK: - Default Methods

override func viewDidLoad() {
    super.viewDidLoad()

    let cellNib = UINib(nibName: "BatchConsumptionCell", bundle: nil)
    tableView.register(cellNib, forCellReuseIdentifier: "BatchConsumptionCell")

    updateTableFields(selectedConsumption: consumption)
}


// MARK: - UI Update Methods

func updateTableFields(selectedConsumption: Consumption) {

    labelId.text = selectedConsumption.wtId
    ...
}

// MARK: - BatchAddEditDelegate Methods

func didFinishEditing(consumption: Consumption) {
    updateTableFields(selectedConsumption: consumption)
}

// MARK: - TableView Methods

override func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? {
    ...
}

override func tableView(_ tableView: UITableView, heightForHeaderInSection section: Int) -> CGFloat {
    ...
}

override func tableView(_ tableView: UITableView, heightForFooterInSection section: Int) -> CGFloat {
    ...
}

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

override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    if indexPath.section == 1 && indexPath.row == 0 {
        let batchConsumptionCell = tableView.dequeueReusableCell(withIdentifier: "BatchConsumptionCell", for: indexPath) as! BatchConsumptionCell
        batchConsumptionCell.setConsumption(consumption: consumption)
        return batchConsumptionCell
    }
    return super.tableView(tableView, cellForRowAt: indexPath)
}


// MARK: - Navigation

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    ...
}

}

BatchConsumptionCell

import UIKit

class BatchConsumptionCell: UITableViewCell, UITableViewDataSource, UITableViewDelegate {

var consumption: Consumption!
var batchConsumptionCount: Int!

@IBOutlet weak var tableView: UITableView!

override func awakeFromNib() {
    super.awakeFromNib()

    var cellNib = UINib(nibName: "BatchConsumptionBasicCell", bundle: nil)
    tableView.register(cellNib, forCellReuseIdentifier: "BatchConsumptionBasicCell")

    cellNib = UINib(nibName: "BatchConsumptionMultiCell", bundle: nil)
    tableView.register(cellNib, forCellReuseIdentifier: "BatchConsumptionMultiCell")

    self.tableView.delegate = self
    self.tableView.dataSource = self
}

func setConsumption(consumption: Consumption) {
    self.consumption = consumption
    self.batchConsumptionCount = consumption.batchConsumptions?.count
}

func numberOfSections(in tableView: UITableView) -> Int {
    return 1
}

func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    if batchConsumptionCount == 1 { // <-- This does not work
        return 3
    } else {
        return batchConsumptionCount
    }
}

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    if batchConsumptionCount == 1 { // <-- This works fine
        let batchConsumptionBasicCell = tableView.dequeueReusableCell(withIdentifier: "BatchConsumptionBasicCell", for: indexPath) as! BatchConsumptionBasicCell
        let bc = consumption.batchConsumptions?.allObjects[0] as! BatchConsumption
        if indexPath.row == 0 {
            batchConsumptionBasicCell.configure(title: "Batch", detail: (bc.batch?.wtId)!)
        } else if indexPath.row == 1 {
            batchConsumptionBasicCell.configure(title: "Weight", detail: String(describing: bc.weight!))
        } else if indexPath.row == 2 {
            batchConsumptionBasicCell.configure(title: "Price", detail: String(format:"%.2f", bc.price))
        }
    } else if batchConsumptionCount >= 2 {
        let batchConsumptionMultiCell = tableView.dequeueReusableCell(withIdentifier: "BatchConsumptionMultiCell", for: indexPath) as! BatchConsumptionMultiCell
        return batchConsumptionMultiCell
    }
    return UITableViewCell()
}

}

查看评论// <-- This works fine
在BatchConsumptionCell

中// <-- This does not work

1 个答案:

答案 0 :(得分:0)

您还应该将nib注册到BatchConsumptionCell类,并实现必要的表视图的数据源&amp;委托方法:

class BatchConsumptionCell: UITableViewCell, UITableViewDataSource, UITableViewDelegate {

    @IBOutlet weak var tableView: UITableView!

    var consumption: Consumption!
    var batchConsumptionCount: Int!

    override func awakeFromNib() {
        super.awakeFromNib()

        var cellNib = UINib(nibName: "BatchConsumptionBasicCell", bundle: nil)
        tableView.register(cellNib, forCellReuseIdentifier: "BatchConsumptionBasicCell")

        cellNib = UINib(nibName: "BatchConsumptionMultiCell", bundle: nil)
        tableView.register(cellNib, forCellReuseIdentifier: "BatchConsumptionMultiCell")

        self.tableView.delegate = self
        self.tableView.dataSource = self
    }

    func setConsumption(consumption: Consumption) {
        self.consumption = consumption
        self.batchConsumptionCount = consumption.batchConsumptions?.count
        tableView.reloadData()
    }

    func numberOfSections(in tableView: UITableView) -> Int {
        return consumption == nil ? 0 : 1
    }

    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        if batchConsumptionCount == .some(1) { return 3 }
        return batchConsumptionCount ?? 0
    }

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

        switch batchConsumptionCount {
        case .some(1):
            let batchConsumptionBasicCell = tableView.dequeueReusableCell(withIdentifier: "BatchConsumptionBasicCell", for: indexPath) as! BatchConsumptionBasicCell
            let bc = consumption.batchConsumptions?.allObjects[0] as! BatchConsumption
            if indexPath.row == 0 {
                batchConsumptionBasicCell.configure(title: "Batch", detail: (bc.batch?.wtId)!)
            } else if indexPath.row == 1 {
                batchConsumptionBasicCell.configure(title: "Weight", detail: String(describing: bc.weight!))
            } else if indexPath.row == 2 {
                batchConsumptionBasicCell.configure(title: "Price", detail: String(format:"%.2f", bc.price))
            }
            return batchConsumptionBasicCell

        default:
            let batchConsumptionMultiCell = tableView.dequeueReusableCell(withIdentifier: "BatchConsumptionMultiCell", for: indexPath) as! BatchConsumptionMultiCell
            return batchConsumptionMultiCell

        }
    }

}