[英]SwiftUI update view on core data object change
我遇到的问题是,当我从工作表视图更新核心数据资产 object 时,更改不会反映在 AssetListView 的 UI 中。 (请注意,从工作表视图中插入新的 object会刷新 AssetListView 的 UI。在工作表视图中删除 object也会刷新AssetListView 的 UI)唯一不起作用的是更新。
当核心数据 object 发生变化时,如何使 AssetListView 发生变化?
我有以下 SwiftUI 代码显示来自 CoreData FetchRequest 的资产列表:
struct AssetListView: View {
@State private var showingSheet = false
@State private var selectedAssetId: NSManagedObjectID?
@Environment(\.managedObjectContext) var moc
@FetchRequest(entity: Asset.entity(), sortDescriptors: [NSSortDescriptor(keyPath: \Asset.allocationPercentage, ascending: false)]) var assets: FetchedResults<Asset>
var body: some View {
VStack {
Form {
ForEach(assets, id: \.self) { asset in
Section {
AssetRowView(asset: asset)
.onTapGesture {
self.selectedAssetId = asset.objectID
self.showingSheet = true
}
}
}
}
}
.navigationBarTitle("Assets").sheet(isPresented: $showingSheet ) {
EditAssetView(assetId: self.selectedAssetId!)
.environment(\.managedObjectContext, self.moc)
}
}
}
}
这是一个编辑屏幕,我将其显示为 SwiftUI 表:
struct EditAssetView: View {
var assetId: NSManagedObjectID
@Environment(\.presentationMode) var presentationMode
@State private var name = ""
@State private var description = ""
@Environment(\.managedObjectContext) var moc
var asset: Asset {
moc.object(with: assetId) as! Asset
}
var body: some View {
NavigationView {
Form {
Section {
TextField("Name", text: $name)
TextField("Description", text: $description)
}
}
.navigationBarTitle(Text("Edit Asset"), displayMode: .inline)
.navigationBarItems(leading: Button("Cancel") {
self.presentationMode.wrappedValue.dismiss()
}, trailing: Button("Done") {
self.presentationMode.wrappedValue.dismiss()
self.asset.name = self.name
self.asset.assetDescription = self.description
try? self.moc.save()
}
)
}
.onAppear {
self.name = self.asset.name
self.description = self.asset.assetDescription
}
}
}
这是 AssetRowView 的代码:
struct AssetRowView: View {
var asset: Asset?
var body: some View {
HStack {
Text(asset.name)
Text(asset.assetDescription)
}
}
}
尝试观察Asset
(因为NSManagedObject
is-a ObservableObject
)
struct AssetRowView: View {
@ObservedObject var asset: Asset // << here !!
var body: some View {
HStack {
Text(asset.name)
Text(asset.assetDescription)
}
}
}
如果这还不够,还可能需要将Done
按钮操作更新为
}, trailing: Button("Done") {
self.asset.objectWillChange.send() // << here !!
self.asset.name = self.name
self.asset.assetDescription = self.description
try? self.moc.save()
// better do this at the end of activity
self.presentationMode.wrappedValue.dismiss()
}
我所做的是将核心数据 object 传递为:
@ObservedObject var toDo: FetchedResults<ToDoModelCoreData>.Element
从我的抓取结果
@FetchRequest var todoFetchRequest: FetchedResults<ToDoModelCoreData>
这将确保 swiftUI 视图在 coredata 为
将@ObservedObject
添加到核心数据 object 有效,但您可能需要滚动列表才能看到更改。 取决于您如何设置谓词、托管对象和获取请求。
在 NSManagedObject 编辑的任何过程结束时添加foo.objectWillChange.send()
可以解决此问题。
就我而言,我需要在编辑工作表中的项目后触发对列表视图的更新。 解决方案是在 DidDismiss 处理程序中调用父 ManagedObject 上的 objectWillChange.send() 方法。
.sheet(item: $selectedChildItem, onDismiss: editViewDidDismiss) { childItem in
EditItemDetailsView(item: childItem)
}
private func editViewDidDismiss(){
self.parentObject.objectWillChange.send()
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.