如何在 SwiftUI 中使用 ForEach 仅更改一个核心数据项的切换?

How to change toggle on just one Core Data item using ForEach in SwiftUI?

如何在没有子视图的情况下只更改列表中的一个开关?如果我从 ForEach 中的所有内容中提取子视图,我知道如何使其工作,但是如何在一个视图上执行它?

我不能使用子视图,因为如果我想从这个子视图中删除一个项目,我以后会遇到问题。它给了我一些我不知道如何修复的错误,所以我试图在一个没有这个错误的视图上进行修复。

列表的代码非常简单:

import SwiftUI

struct ContentView: View {
    
    @Environment(\.managedObjectContext) var moc
    var fetchRequest: FetchRequest<Item>
    var items: FetchedResults<Item> { fetchRequest.wrappedValue }
    
    @State private var doneStatus : Bool = false
    
    var body: some View {
        
        NavigationView {
            List {
                ForEach(items, id: \.self) {item in
                    HStack {
                        Text("\(item.name ?? "default item name")")
                        
                        Spacer()
                        
                        Toggle(isOn: self.$doneStatus) {
                            Text("Done")
                        }
                        .labelsHidden()
                        .onAppear {
                            self.doneStatus = item.done
                        }
                        .onTapGesture {
                            self.doneStatus.toggle()
                            item.done.toggle()
                            try? self.moc.save()
                        }
                    }
                }
                .onDelete(perform: removeItem)
            }
            .navigationBarTitle("Items")
            .navigationBarItems(
                leading:
                Button(action: {
                    for number in 1...3 {
                        let item = Item(context: self.moc)
                        item.date = Date()
                        item.name = "Item \(number)"
                        item.done = false
                        
                        do {
                            try self.moc.save()
                        }catch{
                            print(error)
                        }
                    }
                }) {
                    Text("Add 3 items")
                }
            )
        }
    }
    
    init() {
        fetchRequest = FetchRequest<Item>(entity: Item.entity(), sortDescriptors: [
            NSSortDescriptor(keyPath: \Item.name, ascending: true)
        ])
    }
    
    func removeItem(at offsets: IndexSet) {
        for offset in offsets {
            let item = items[offset]
            moc.delete(item)
        }
        try? moc.save()
    }
}

struct ContentView_Previews: PreviewProvider {
    static var previews: some View {
        let context = (UIApplication.shared.delegate as! AppDelegate).persistentContainer.viewContext
        //Test data
        let testItem = Item.init(context: context)
        testItem.date = Date()
        testItem.name = "Item name"
        testItem.done = false
        return ContentView().environment(\.managedObjectContext, context)
    }
}

我正在使用 1 个核心数据实体:项目。具有3个属性:日期(Date)、完成(Boolean)、名称(String)。

问题

当我点击一个开关时,所有其他开关也会发生变化。

我找不到使用 Core Data 的解决方案。我想也许我应该使用 .id 而不是 .self?并向我的实体添加另一个属性:id (UUID)。但是我尝试去做,但失败了。

如有任何帮助,我将不胜感激。

您将所有Toggle绑定到一个状态...所以

  1. 删除这个
//    @State private var doneStatus : Bool = false
  1. Toggle 动态绑定到当前迭代 item(注意:不再需要 .onAppear/.onTapGesture)
Toggle(isOn: Binding<Bool>(
    get: { item.done },
    set: {
        item.done = [=11=]
        try? self.moc.save()
    })) {
    Text()
}
.labelsHidden()

backup