我正在使用Swift 4开发一个应用。我正在以编程方式构建我的应用 -没有情节提要。
我有一个看起来像这样的ViewController
class SearchController: UITableViewController {
private let cellID = "cellID"
fileprivate let searchController = UISearchController(searchResultsController: nil)
fileprivate let viewModel = SearchViewModel(client: DiscogClient())
override func viewDidLoad() {
super.viewDidLoad()
setupSearchBar()
setupTableView()
}
fileprivate func setupSearchBar() -> Void {
navigationItem.hidesSearchBarWhenScrolling = false
navigationItem.searchController = searchController
searchController.searchBar.delegate = self
}
fileprivate func setupTableView() -> Void {
tableView.register(UITableViewCell.self, forCellReuseIdentifier: cellID)
tableView.tableFooterView = UIView()
}
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return viewModel.cellViewModels.count
}
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: cellID, for: indexPath)
return cell
}
}
extension SearchController: UISearchBarDelegate {
func searchBar(_ searchBar: UISearchBar, textDidChange searchText: String) {
viewModel.search(term: searchText)
}
}
我有一个看起来像这样的ViewModel
import Foundation
import UIKit
class SearchViewModel {
var cellViewModels: [Result] = []
private let client: APIClient
var results: SearchResults? {
didSet {
for result in (results?.results)! {
cellViewModels.append(result)
}
}
}
var isLoading: Bool = false {
didSet {
showLoading?()
}
}
var showLoading: (() -> Void)?
var reloadData: (() -> Void)?
var showError: ((Error) -> Void)?
init(client: APIClient) {
self.client = client
}
func search(term: String) {
if let client = client as? DiscogClient {
let endpoint = DiscogsEndpoint.search(term: term, searchType: .release)
client.fetch(with: endpoint) { (either) in
switch either {
case .success(let results):
self.results = results
case .error(let error):
self.showError?(error)
}
}
}
}
}
当我在didSet
上检查var results: SearchResults?
时,可以看到该值是通过我的APIClient
设置的。
我要做的就是指示ViewController表在设置该值后重新加载其数据。
我不确定如何实现此目标?我以为我可以在cellViewModels上使用didSet
并扩展到ViewController上的方法。这行不通,或者我不知道如何执行此操作。
我现在希望看到行数通过我的SearchController中的viewModel.cellViewModels.count
更新
答案 0 :(得分:0)
您可以修改您的search
以接受完成阻止
func search(term: String, @escaping completion :() -> ()) {
if let client = client as? DiscogClient {
let endpoint = DiscogsEndpoint.search(term: term, searchType: .release)
client.fetch(with: endpoint) { (either) in
switch either {
case .success(let results):
self.results = results
completion()
case .error(let error):
self.showError?(error)
completion()
}
}
}
}
最后,您可以重新加载tableView
self.search(term: "abcd") {
self.tableView.reloadData()
}
人们可能经常要求您implement protocols
并在ViewModel和View之间建立通信。但是,使用块还是协议始终是设计实现的详细信息:)
希望这会有所帮助。