SwiftUI 和核心数据 - 如何在谓词中使用父记录?

rad*_*dev 5 core-data swiftui

是否可以在 SwiftUI 的 @FetchRequest 属性包装器内的谓词中使用核心数据记录?

我有一个项目列表和一个任务列表。我想点击一个项目并导航到该项目的相关任务列表。我似乎无法找到一种在初始化 @FetcheRequest之前以 SwiftUI可以看到的方式传入父项目的方法。

我尝试将父项目放在 EnvironmentObject 中。当我从 ProjectListView 导航到 TaskListView 时会调用它。

TaskListView()
   .environment(\.managedObjectContext, self.managedObjectContext)
   .environmentObject(self.projectToEdit)
Run Code Online (Sandbox Code Playgroud)

然后在我添加的 TaskListView 中尝试了这个:

    @Environment(\.managedObjectContext) var managedObjectContext

    @EnvironmentObject var parentProject: Project

    @FetchRequest(
        entity: Task.entity(),
        sortDescriptors: [
            NSSortDescriptor(keyPath: \Task.name, ascending: true)
        ],
        predicate: NSPredicate(format: String(format: "%@%@", "taskProject", " == %@"), parentProject)
    ) var tasks: FetchedResults<Task>
Run Code Online (Sandbox Code Playgroud)

我在谓词的行上收到以下错误。

不能在属性初始值设定项中使用实例成员“parentProject”;属性初始值设定项在 'self' 可用之前运行

那么有没有办法以某种方式编写一个可以使用父项目的谓词?将项目传递给任务视图似乎行不通。我还能如何在这样的谓词中使用记录?

小智 5

FetchRequest 可以在init方法中动态创建。这样你就可以改变谓词和排序条件。这是一些示例代码来实现这一点。

// sample Project class
class Project:NSManagedObject {
    var id : String
    var name : String
}

// sample Task class
class Task:NSManagedObject {
    var id : String
    var prjId : String
    var name : String
}

// Task List View
struct TaskListView: View {
    @Environment(\.managedObjectContext) var managedObjectContext

    private var tasksRequest: FetchRequest<Task>
    private var tasks: FetchedResults<Task> { tasksRequest.wrappedValue }

    private var project:Project

    // init Task with Project
    init(_ project:Project) {
        self.project = project

        // create FetchRequest
        self.tasksRequest = FetchRequest(
            entity: Task.entity(),
            sortDescriptors: [NSSortDescriptor(key: "name", ascending:true)],
            predicate: NSPredicate(format: "prjId == %@", project.id))
    }

    var body: some View {
        VStack {
            Section(header: Text("Tasks under \(project.name):")) {
                // access the fetched objects
                ForEach(tasks, id:\.id) { task in
                    Text("\(task.name)")
                }
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

然后调用TaskListView()将如下所示:

// call to TaskListView
TaskListView(self.projectToEdit)
    .environment(\.managedObjectContext, self.managedObjectContext)
Run Code Online (Sandbox Code Playgroud)