如何将 TableView 放入 CollectionViewCell 中?

How can l put TableView inside CollectionViewCell?

我有一个包含多个单元格的 tableView(由 MVVM 架构创建)。

在 ViewController 中,我这样填充我的 tableView:

tbv.registerCells(
        withModels:
        FirstViewModel.self,
        SecondViewModel.self,
        ThirdViewModel.self)

我的任务是将我的 TableView 放在 CollectionView 的一个单元格中。我认为我必须在创建 CollectionViewCell 和 CollectionCellViewModel 之后在我的 ViewController 中创建 CollectionView,但是我不明白具体怎么做。

如果你知道,怎么做,帮助。

在 Tableview 中,您可以通过传递 collectionviewdata

加载 UITableViewCell
//View Controller

var collectionView1Data = ["cell1", "cell2"]
var collectionView2Data = ["cell1", "cell2"]

//UITableviewDelegate Datasource
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    //.....
    if indexPath.row == 1 {
        let cell = tableView.dequeueReusableCell(withIdentifier: "CellID") as? TableviewCell
        cell.collectionData = collectionView1Data /// Collectionviewdata 
        return cell
    } else {
        let cell = tableView.dequeueReusableCell(withIdentifier: "CellID") as? TableviewCell
        cell.collectionData = collectionView2Data
        return cell
    }
}

================================
每个 Tableviewcell 包含 CollectionView

//UITableViewCell
class TableviewCell: UITableViewCell {
    @IBOutlet weak var collectionView: UICollectionView!
    var collectionData: [String]? {
        didSet {
            guard collectionData != nil else {
                return
            }
            collectionView.reloadData()
        }
    }
    override func awakeFromNib() {
        super.awakeFromNib()
        collectionView.register(UINib(nibName: "collectionViewCell", bundle: nil), forCellWithReuseIdentifier: "collectionViewCell")
        collectionView.dataSource = self
        collectionView.delegate = self
    }
}

extension TableviewCell: UICollectionViewDataSource, UICollectionViewDelegate,UICollectionViewDelegateFlowLayout {
    func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
        collectionData.count ?? 0
    }
    
    func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
        let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "collectionViewCell", for: indexPath) as? collectionViewCell
        cell...
        return cell
    }
}

我如何在我的一个应用程序的集合视图中有多个 table 视图。首先,我有一个视图控制器,我在其中构建我的集合视图。正如新设计指南中通常建议的那样,我在这个视图控制器的扩展中有 Collection View 委托和数据源。

在您的视图控制器中,您为 table 视图定义了一个委托和数据源。最好,这是一个不同的class。我不会将 table 视图数据源和委托也放在与您的集合视图相同的视图控制器中。

class WorkoutSettingsViewController: UIViewController, LoadWorkoutSettings {

    //MARK: - Properties
    //Used variables

    //Used constants
    private let settingsDelegate = SettingsTableViewDelegate()

扩展程序将如下所示。

extension WorkoutSettingsViewController: UICollectionViewDataSource, UICollectionViewDelegate {
    
    func numberOfSections(in collectionView: UICollectionView) -> Int {
        //Whatever sets your sections
    }
    
    func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
        //Whatever sets your rows per section
    }

    func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
        
        let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "Workout Settings", for: indexPath) as! SettingsCollectionViewCell
            
        settingsDelegate.workoutTitleLabel = [countdown, mainView, spokenMessage]
        settingsDelegate.mainContentLabel = getSettingsContent()
        cell.settingsTableView.delegate = settingsDelegate
        cell.settingsTableView.dataSource = settingsDelegate
        cell.settingsTableView.reloadData()

        return cell
    }
    
    func collectionView(_ collectionView: UICollectionView, viewForSupplementaryElementOfKind kind: String, at indexPath: IndexPath) -> UICollectionReusableView {
        
     //Whatever you need as header or footer
}

委托完全按照您希望 table 视图数据源和委托执行的操作执行。

class SettingsTableViewDelegate: NSObject, UITableViewDataSource, UITableViewDelegate {
    
    //MARK: - Properties
    //Used variables
    var workoutTitleLabel = [String]()
    var mainContentLabel = [String]()
    var selectedSetting: ((Int) -> ())? = .none
    private var secondaryContentLabel = [String]()
    
    //Used constants
    private let onTitle = NSLocalizedString("ON", comment: "Localized on title")
    private let offTitle = NSLocalizedString("OFF", comment: "Localized off title")
    private let fontColorBlack = UIColor(red: 20.0/255.0, green: 20.0/255.0, blue: 19.0/255.0, alpha: 1.0)
    private let fontColorRed = UIColor(red: 255.0/255.0, green: 96.0/255.0, blue: 89.0/255.0, alpha: 1.0)
    
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        workoutTitleLabel.count
    }
    
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        
        let cell = tableView.dequeueReusableCell(withIdentifier: "Settings Cell") as! WorkoutTableViewCell
        
        cell.workoutTitle.text = workoutTitleLabel[indexPath.row]
        cell.mainContent.text = mainContentLabel[indexPath.row]
        cell.secondaryContent.text = ""
        
        (mainContentLabel[indexPath.row] == offTitle) ? (cell.mainContent.textColor = fontColorRed) : (cell.mainContent.textColor = fontColorBlack)
        
        return cell
    }
    
    func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
        tableView.deselectRow(at: indexPath, animated: true)
        selectedSetting?(indexPath.row)
    }
    
    func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
        61
    }
}

您的集合视图单元格应如下所示。

class SettingsCollectionViewCell: UICollectionViewCell {
    
    @IBOutlet weak var settingsTableView: UITableView!
}

这应该可以工作。如果您需要从 table 视图委托/数据源回调到管理您的集合视图的视图控制器,您可以使用闭包。在示例 table 视图委托中,闭包称为 selectedSettings。在 viewDidLoad 的视图控制器中,您可以像这样定义回调:

override func viewDidLoad() {
    super.viewDidLoad()
    
    settingsDelegate.selectedSetting = { [unowned self] selection in
        startSettingsMenu(for: selection)
    }
    
}

结果是这样的。

亲切的问候, MacUserT