Select 和删除 macOS 上 SwiftUI 列表视图中的核心数据实体

Select and Delete Core Data entities in SwiftUI List View on macOS

我是 SwiftUI 的新手,但取得了一定的进步。我使用的是最新版本的 Xcode 12.4 和 运行ning BigSur 11.2.1。我正处于想要使用核心数据但 运行 无法找到修复的问题的阶段。

当我创建基本 Xcode 项目时,我 select App 和 macOS 作为模板 那我select 界面 - SwiftUI, 生命周期 - SwiftUI 应用程序, 语言 - Swift 和 select 使用核心数据

一个新项目被创建并且构建和运行没有任何问题。 在出现的 window 中,我只需单击顶部栏上的 + 按钮即可添加新项目(日期戳)。到目前为止,一切都很好。这都是香草苹果代码。

我被卡住的地方:- 列表 - ContentView 中的 ForEach 视图不允许通过单击 select 编辑任何实体(项目),因此我找不到方法来删除条目。

如果我用一组文本项替换实体,那么我可以 select 它们并删除它们 通过使用 @State var selectKeeper = Set() 使用 selection:列表视图中的 $selectKeeper。

有人可以解释一下怎么做吗?

这是内容视图的原始代码。

import SwiftUI
import CoreData

struct ContentView: View {
    @Environment(\.managedObjectContext) private var viewContext

    @FetchRequest(
        sortDescriptors: [NSSortDescriptor(keyPath: \Item.timestamp, ascending: true)],
        animation: .default)
    private var items: FetchedResults<Item>

    var body: some View {
        List {
            ForEach(items) { item in
                Text("Item at \(item.timestamp!, formatter: itemFormatter)")
            }
            .onDelete(perform: deleteItems)
        }
        .toolbar {
            Button(action: addItem) {
                Label("Add Item", systemImage: "plus")
            }
        }
    }

    private func addItem() {
        withAnimation {
            let newItem = Item(context: viewContext)
            newItem.timestamp = Date()

            do {
                try viewContext.save()
            } catch {
                // Replace this implementation with code to handle the error appropriately.
                // fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
                let nsError = error as NSError
                fatalError("Unresolved error \(nsError), \(nsError.userInfo)")
            }
        }
    }

    private func deleteItems(offsets: IndexSet) {
        withAnimation {
            offsets.map { items[[=11=]] }.forEach(viewContext.delete)

            do {
                try viewContext.save()
            } catch {
                // Replace this implementation with code to handle the error appropriately.
                // fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
                let nsError = error as NSError
                fatalError("Unresolved error \(nsError), \(nsError.userInfo)")
            }
        }
    }
}

private let itemFormatter: DateFormatter = {
    let formatter = DateFormatter()
    formatter.dateStyle = .short
    formatter.timeStyle = .medium
    return formatter
}()

struct ContentView_Previews: PreviewProvider {
    static var previews: some View {
        ContentView().environment(\.managedObjectContext, PersistenceController.preview.container.viewContext)
    }
}

您应该添加 EditButton() 并可能将所有这些包装在 NavitagionView 中可能会为您提供您正在寻找的内容:

var body: some View {
    NavigationView{
        List {
            ForEach(items) { item in
                Text("Item at \(item.timestamp!, formatter: itemFormatter)")
            }
            .onDelete(perform: deleteItems)
        }
        .toolbar {
            ToolbarItem(placement: .navigationBarLeading) {
                #if os(iOS)
                EditButton()
                #endif
            }

            ToolbarItem(placement: .navigationBarTrailing) {
                Button(action: addItem) {
                Label("Add Item", systemImage: "plus")
                }
            }
        }
    }
}