如何解决@published 不适用于类型数组,但适用于单一类型
我正在测试非常简单的代码,有一个类确认 @ObservableObject
协议,并且有两个 @Published
属性,一个是我的自定义 type
对象数组,另一个是只是同班的普通 type
。我有两种观点,其中 parent 拥有单一的事实来源,它有一个 List
和一个 navigationLink
。 NavigationLink 将 @Published
数组对象作为 Binding
传递给 subView。
问题-:
当我输入 TextField
时,视图没有更新。它正在修改 @Published
对象属性,但不更新子视图。这仅在我使用 Array
个对象进行测试时发生,但现在如果我注释 array
个对象属性并简单地使用另一个(没有数组),它会更新得很好。
谁能解释一下背后的原因,这是一个错误吗?还是 Array
在幕后做其他事情?
不工作的例子-:
import SwiftUI
struct Person: Identifiable {
let id: String = UUID().uuidString
var name: String = "Tushar"
var age: Int = 25
init(name: String,age: Int) {
self.name = name
self.age = age
}
}
class PersonStore: ObservableObject {
@Published var name: [Person] = [Person(name:"foo",age:23),Person(name:"foobar",age:23)]
//@Published var name: Person = Person(name:"foo",age:23)
}
struct ContentViewsss: View {
@Observedobject var store = PersonStore()
@State var isEditing = false
var body: some View {
NavigationView{
List(store.name.indices){ index in
NavigationLink(destination: ContentViewsss1(store: $store.name[index])) {
Text("Edit")
}
}.navigationTitle(“Welcome")
}
}
}
struct ContentViewsss1: View {
@Binding var store: Person
var body: some View {
TextField("My Text",text: $store.name)
Text(store.name)
}
}
工作示例-:
import SwiftUI
struct Person: Identifiable {
let id: String = UUID().uuidString
var name: String = "Tushar"
var age: Int = 25
init(name: String,age: Int) {
self.name = name
self.age = age
}
}
class PersonStore: ObservableObject {
// @Published var name: [Person] = [Person(name:"foo",age:23)]
@Published var name: Person = Person(name:"foo",age:23)
}
struct ContentViewsss: View {
@Observedobject var store = PersonStore()
@State var isEditing = false
var body: some View {
NavigationView{
List{
NavigationLink(destination: ContentViewsss1(store: $store.name)) {
Text("Edit")
}
}.navigationTitle("Welcome")
}
}
}
struct ContentViewsss1: View {
@Binding var store: Person
var body: some View {
TextField("My Text",text: $store.name)
Text(store.name)
}
}
解决方法
据说,导致问题的不是数组——而是List
。您会在原始代码中看到,如果将 List
更改为 ForEach
,它将起作用。我不完全确定故障发生在哪里,而且它比人们想象的更微妙。
我的建议是使用中间视图模型。以下代码有效:
class ViewModel: ObservableObject {
@Published var person : Person {
didSet {
binding.wrappedValue = person
}
}
var binding : Binding<Person>
init(withPersonBinding person: Binding<Person>) {
self.person = person.wrappedValue
self.binding = person
}
}
struct ContentViewsss1: View {
//@Binding var store: Person
@ObservedObject var store : ViewModel
var body: some View {
TextField("My Text",text: $store.person.name)
Text(store.person.name)
}
}
被这样调用:
NavigationLink(destination: ContentViewsss1(store: ViewModel(withPersonBinding: $store.name[index]))) {
Text("Edit")
}
现在,这是非常奇怪的一点。如果您返回原始代码并将详细信息视图修改为:
struct ContentViewsss1: View {
@Binding var store: Person
@State private var txt = ""
var body: some View {
TextField("My Text",text: $store.name)
Text(store.name)
TextField("Other text",text: $txt)
}
}
您会看到,如果您编辑第一个文本字段,您的更改将不会得到反映(正如您所经历的那样)。 但是,只要您更改了第二个字段中的文本,@State 中的更改就会触发它也检查@Binding,并且该字段会更新。同样,这仅发生在 List
而不是 ForEach
。 SwiftUI 的奥秘...
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。