当您离开并返回时,SwiftUI FetchedResult视图将无法更新。

我有一个简单的待办事项清单应用程序作为示例创建。该应用包含2个实体:


一个TodoList,其中可以包含许多TodoItem
一个TodoItem,属于一个TodoList


首先,这是我的核心数据模型:

ios - 离开 View 并返回后,FetchedResult View 不会更新-LMLPHP

ios - 离开 View 并返回后,FetchedResult View 不会更新-LMLPHP

对于实体,我在Class Definition中使用CodeGen

在此示例中,我仅使用4个小视图。

TodoListView

struct TodoListView: View {
    @Environment(\.managedObjectContext) var managedObjectContext
    @FetchRequest(
        entity: TodoList.entity(),
        sortDescriptors: []
    ) var todoLists: FetchedResults<TodoList>
    @State var todoListAdd: Bool = false

    var body: some View {
        NavigationView {
            List {
                ForEach(todoLists, id: \.self) { todoList in
                    NavigationLink(destination: TodoItemView(todoList: todoList), label: {
                        Text(todoList.title ?? "")
                    })
                }
            }
            .navigationBarTitle("Todo Lists")
            .navigationBarItems(trailing:
                Button(action: {
                    self.todoListAdd.toggle()
                }, label: {
                    Text("Add")
                })
                .sheet(isPresented: $todoListAdd, content: {
                    TodoListAdd().environment(\.managedObjectContext, self.managedObjectContext)
                })
            )
        }
    }
}


这只是获取所有TodoList,并将它们吐出列表。导航栏中有一个按钮,可用于添加新的待办事项列表。

TodoListAdd

struct TodoListAdd: View {
    @Environment(\.presentationMode) var presentationMode
    @Environment(\.managedObjectContext) var managedObjectContext
    @State var todoListTitle: String = ""

    var body: some View {
        NavigationView {
            Form {
                TextField("Title", text: $todoListTitle)

                Button(action: {
                    self.saveTodoList()
                    self.presentationMode.wrappedValue.dismiss()
                }, label: {
                    Text("Save")
                })

                Button(action: {
                    self.presentationMode.wrappedValue.dismiss()
                }, label: {
                    Text("Cancel")
                })
            }
            .navigationBarTitle("Add Todo List")
        }
        .navigationViewStyle(StackNavigationViewStyle())
    }

    func saveTodoList() {
        let todoList = TodoList(context: managedObjectContext)
        todoList.title = todoListTitle

        do { try managedObjectContext.save() }
        catch { print(error) }
    }
}


这只是保存了一个新的待办事项清单,然后关闭了模态。

TodoItemView

struct TodoItemView: View {
    @Environment(\.managedObjectContext) var managedObjectContext
    var todoList: TodoList
    @FetchRequest var todoItems: FetchedResults<TodoItem>
    @State var todoItemAdd: Bool = false

    init(todoList: TodoList) {
        self.todoList = todoList
        self._todoItems = FetchRequest(
            entity: TodoItem.entity(),
            sortDescriptors: [],
            predicate: NSPredicate(format: "todoList == %@", todoList)
        )
    }

    var body: some View {
        List {
            ForEach(todoItems, id: \.self) { todoItem in
                Button(action: {
                    self.checkTodoItem(todoItem: todoItem)
                }, label: {
                    HStack {
                        Image(systemName: todoItem.checked ? "checkmark.circle" : "circle")
                        Text(todoItem.title ?? "")
                    }
                })
            }
        }
        .navigationBarTitle(todoList.title ?? "")
        .navigationBarItems(trailing:
            Button(action: {
                self.todoItemAdd.toggle()
            }, label: {
                Text("Add")
            })
            .sheet(isPresented: $todoItemAdd, content: {
                TodoItemAdd(todoList: self.todoList).environment(\.managedObjectContext, self.managedObjectContext)
            })
        )
    }

    func checkTodoItem(todoItem: TodoItem) {
        todoItem.checked = !todoItem.checked

        do { try managedObjectContext.save() }
        catch { print(error) }
    }
}


该视图获取属于已点击的TodoList的所有TodoItem。这是发生问题的地方。我不确定是否是因为在这里使用了init(),但是有一个错误。首次进入此视图时,可以点击待办事项以“检查”该更改,更改立即显示在视图中。但是,当您导航到另一个TodoList的另一个TodoItemView并返回时,点击该视图时将不再更新。选中标记图像未显示,您需要离开该视图,然后重新输入该视图,以使上述更改实际出现。

TodoItemAdd

struct TodoItemAdd: View {
    @Environment(\.presentationMode) var presentationMode
    @Environment(\.managedObjectContext) var managedObjectContext
    var todoList: TodoList
    @State var todoItemTitle: String = ""

    var body: some View {
        NavigationView {
            Form {
                TextField("Title", text: $todoItemTitle)

                Button(action: {
                    self.saveTodoItem()
                    self.presentationMode.wrappedValue.dismiss()
                }, label: {
                    Text("Save")
                })

                Button(action: {
                    self.presentationMode.wrappedValue.dismiss()
                }, label: {
                    Text("Cancel")
                })
            }
            .navigationBarTitle("Add Todo Item")
        }
        .navigationViewStyle(StackNavigationViewStyle())
    }

    func saveTodoItem() {
        let todoItem = TodoItem(context: managedObjectContext)
        todoItem.title = todoItemTitle
        todoItem.todoList = todoList

        do { try managedObjectContext.save() }
        catch { print(error) }
    }
}


这仅允许用户添加新的待办事项。

如前所述,当您离开并重新进入TodoItemView时,视图将自动停止更新。这是此行为的记录:

https://i.imgur.com/q3ceNb1.mp4

我到底在做什么错?如果由于导航链接中的视图在显示之前就已初始化,则不应该使用init(),那么正确的实现是什么?

最佳答案

经过数小时搜寻该问题的各种不同短语后,找到了解决方案:https://stackoverflow.com/a/58381982/10688806

您必须使用“惰性视图”。

码:

struct LazyView<Content: View>: View {
    let build: () -> Content

    init(_ build: @autoclosure @escaping () -> Content) {
        self.build = build
    }

    var body: Content {
        build()
    }
}


用法:

NavigationLink(destination: LazyView(TodoItemView(todoList: todoList)), label: {
    Text(todoList.title ?? "")
})

关于ios - 离开 View 并返回后,FetchedResult View 不会更新,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/59061371/

10-09 09:59