gpt4 book ai didi

ios - 在 Swift 中将数据传递给 ViewModel

转载 作者:搜寻专家 更新时间:2023-10-31 21:46:11 24 4
gpt4 key购买 nike

我是 MVVM 模式的新手,所以请不要评判我。我想要做的是,如果可能的话,我想将数据与 segue 一起传递给 ViewModel 并基于它填充数据。为什么我要这样做,因为假设我有 3 个 ViewController(A、B、C)。在 ViewController 上,我从 Realm 数据库获取数据,然后使用 segues 传递这些数据,我真的不想更改该结构。但是,如果您有更好的建议,也请告诉我。

这是我的代码:

//MARK:- ViewModelItemTypes
enum EventViewModelItemType {
case description
case materials
}

//MARK:- ViewModel

class EventViewModel: NSObject {
var items = [EventViewModelItem]()

override init() {
super.init()

let confEvent = ConferenceEvents()
let confMaterials = EventMaterials()

if let descriptionText = confEvent.eventDescription {
let descriptionItem = EventViewModelDescItem(descriptionText: descriptionText)
items.append(descriptionItem)
}

if let materials: EventMaterials = confMaterials {
let materialsItem = EventViewModeMaterialsItem(materials: [materials])
items.append(materialsItem)
}
}
}

extension EventViewModel: UITableViewDataSource {
func numberOfSections(in tableView: UITableView) -> Int {
return items.count
}

func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return items[section].rowCount
}

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let item = items[indexPath.row]
switch item.type {
case .description:
if let cell = tableView.dequeueReusableCell(withIdentifier: Identifiers.confEventDescriptionCell, for: indexPath) as? ConfEventDescTCell {
cell.item = item
return cell
}
case .materials:
if let item = item as? EventViewModeMaterialsItem, let cell = tableView.dequeueReusableCell(withIdentifier: Identifiers.confEventMaterialCell, for: indexPath) as? ConfEventMatTCell {
let materials = item.materials[indexPath.row]
cell.material = materials
return cell
}
}
return UITableViewCell()
}

func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? {
return items[section].sectionTitle
}
}

//MARK:- ViewModelItem Protocol
protocol EventViewModelItem {
var type: EventViewModelItemType { get }
var rowCount: Int { get }
var sectionTitle: String { get }
}

extension EventViewModelItem {
var rowCount: Int {
return 1
}
}

//MARK:- View Model Items
class EventViewModelDescItem: EventViewModelItem {
var type: EventViewModelItemType {
return .description
}

var sectionTitle: String {
return "Description"
}

var descriptionText: String

init(descriptionText: String) {
self.descriptionText = descriptionText
}
}

class EventViewModeMaterialsItem: EventViewModelItem {
var type: EventViewModelItemType {
return .materials
}
var sectionTitle: String {
return "Materials"
}
var rowCount: Int {
return materials.count
}
var materials: [EventMaterials]

init(materials: [EventMaterials]) {
self.materials = materials
}
}

如您所见,我有 confEvent 和 confMaterials 的空实例。它们是我的模型,我正在使用 RealmSwift。 我的建议是用数据填充这些模型,但我不知道该怎么做。这就是我希望你们帮助我的地方。

这是我的 ViewController 代码:

class ViewController: UIViewController {

@IBOutlet private weak var tableView: UITableView!

var viewModel = EventViewModel()

override func viewDidLoad() {
super.viewDidLoad()
tableView.sectionHeaderHeight = 70
tableView.separatorStyle = .none
tableView.dataSource = viewModel
tableView?.estimatedRowHeight = 100
tableView.rowHeight = UITableViewAutomaticDimension
tableView.register(ConfEventDescTCell.nib, forCellReuseIdentifier: Identifiers.confEventDescriptionCell)
tableView.register(ConfEventMatTCell.nib, forCellReuseIdentifier: Identifiers.confEventMaterialCell)

}
}

UITableViewCells:第一个 UITableViewCell

import UIKit

class ConfEventDescTCell: UITableViewCell {

@IBOutlet private weak var descriptionTextView: UITextView!

var item: EventViewModelItem? {
didSet {
configureUI()
}
}

private func configureUI() {
guard let item = item as? EventViewModelDescItem else { return }
DispatchQueue.main.async {
self.descriptionTextView.text = item.descriptionText
}
}

static var nib:UINib {
return UINib(nibName: identifier, bundle: nil)
}

static var identifier: String {
return String(describing: self)
}
}

第二个 UITableViewCell

import UIKit

class ConfEventMatTCell: UITableViewCell {

@IBOutlet private weak var matTitle: UIImageView!

var material: EventMaterials? {
didSet {
guard let material = material else { return }
configureUI()

}
}

private func configureUI() {
guard let material = material else { return }
DispatchQueue.main.async {
self.matTitle.text = material.materialTitle
}
}

static var nib: UINib {
return UINib(nibName: identifier, bundle: nil)
}

static var identifier: String {
return String(describing: self)
}
}

最佳答案

您应该将您的 viewModel 传递到创建它的 ConfEventDescTCell 中。

例如,在 cellForRowAtIndexPath 中,您将创建一个新的 UITableViewCell,然后使用方法调用或作为属性或其他方式将 viewmodel 实例传递给它。问题是有很多方法可以做到这一点。

关键是在显示单元格之前,您从特定 View 模型中检索信息以对其进行自定义。

更新:我认为最好不要在您的 View 模型中包含空模型。那里根本不应该有模型。更好的解决方案可能是概括您获取数据的方式,然后在您实际需要它的 View Controller 中获取它。在那里,您可以直接为那里的 View 创建正确的 View 模型。

关于ios - 在 Swift 中将数据传递给 ViewModel,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50852896/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com