我有一个表单,它获取一个邮政编码并将其传递给一个方法。该方法查找城市和州,并将信息作为元组返回。该方法还有一个完成处理程序,以便在找到城市和州之前不会保存表单数据的其余部分。
我把函数移到了视图模型中,以保持事物的有序,但现在我很困惑。首先,必须在函数返回之前调用完成处理程序,尽管这似乎与我想要的完全相反。我想获得城市和州的值,然后我想通过处理程序保存它们。我困惑的后半部分在于函数的调用。如果我分配一个像cityState
这样的变量来从我的方法接收元组,那么保存将发生在一个我无法访问的闭包中!(请参阅下面的设置视图代码。(
我可以把所有的东西都移回它正常工作的地方。但我正在努力理解MVVM应该如何工作。
设置视图模型
func getCityStateFromPostalCode(zip: String, completion: @escaping () -> ()) -> (String, String) {
let geocoder = CLGeocoder()
var city = ""
var state = ""
geocoder.geocodeAddressString(zip) { (placemarks, error) in
if let placemark = placemarks?[0] {
if placemark.postalCode == zip {
city = placemark.locality!
state = placemark.administrativeArea!
}
}
completion()
}
return (city, state)
}
设置查看
let settingsVM = SettingsViewModel()
let cityState = settingsVM.getCityStateFromPostalCode(zip: companyZip) {
let newCompany = Company(context: self.moc)
newCompany.id = UUID()
newCompany.city = ?? //can't use cityState here.
newCompany.state = ?? // or here!
}
在完成处理程序中返回city
和state
,而不是在return
:中
func getCityStateFromPostalCode(zip: String, completion: @escaping ((String, String)) -> ()) {
let geocoder = CLGeocoder()
var city = ""
var state = ""
geocoder.geocodeAddressString(zip) { (placemarks, error) in
if let placemark = placemarks?[0] {
if placemark.postalCode == zip {
city = placemark.locality!
state = placemark.administrativeArea!
}
}
completion((city, state))
}
}
然后你可以做:
settingsVM.getCityStateFromPostalCode(zip: companyZip) { city, state in
let newCompany = Company(context: self.moc)
newCompany.id = UUID()
newCompany.city = city
newCompany.state = state
}
@ObservedObject var model: someViewModel
var body: some View {
ScrollView {
ScrollViewReader { pageScroller in
ForEach(0..<100) { i in
Text("Example (i)")
.frame(width: 200, height: 200)
.background(colors[i % colors.count])
.id(i)
}
//Magic here
.onReceive(model.scroller.objectWillChange) {
pageScroller.scrollTo(model.scroller.scrollToPageName, anchor: .top)
}
}
}
.frame(height: 350)
}
class someViewModel: ObservableObject {
//Magic here
@ObservedObject var scroller: scrollerHelper
init() {
someDelayedCall()
}
func someDelayedCall(){
scroller.scrollToPageName = 8
}
}
//Magic here
class scrollerHelper: ObservableObject {
@Published var scrollToPageName: Int = -1
}
scrollerHelper
的主要思想必须作为ObservedObject在模型内部。
在这种情况下,您可以使用它作为强制代码从您的视图中运行的东西。
这里的神奇之处在于,视图不会像模型的任何@Published
变量发生变化时那样完全刷新。此代码将以与视图中加载的按钮内部代码相同的方式运行!太棒了,对吧?(
但如果您不需要这样的行为,您可以使用swiftUI模型的标准使用方式。我在这里描述过:
https://stackoverflow.com/a/62919526/4423545