Using SwiftUI, Core Data, and one-to-many relationships, why does the list not update when adding a row on the Many side

后端 未结 3 1991
我在风中等你
我在风中等你 2021-01-22 12:18

I have a SwiftUI project with Core Data. The data model is a simple one-to-many and two primary views which each have a textfield at the top and a button to add a new item to th

3条回答
  •  走了就别回头了
    2021-01-22 13:19

    I've found a fix/workaround that's literally just a few lines of code and seems to work great.

    What's happening as you've seen is CoreData isn't announcing anything has changed when a relationship changes (or for that matter a relation to a relation). So your view struct isn't getting reinstantiated and it's not querying those computed properties on your core data object. I've been learning SwiftUI and trying to rewrite a UI to use a Model that uses a few different relationships, some nested.

    My initial thought was to use subviews with @FetchRequests and pass in parameters to those views. But I've got a lot of subviews that need to make use of relationships - that's a ton of code, and for me could potentially be tens if not hundreds of fetchrequests for some layouts. I'm no expert, but that way lies madness!

    Instead I've found a way that seems hackish, but uses very little code and feels kind of elegant for a cheat.

    I have a ModelController class that handles all Core Data code on a background context and I use that context to 'kick' the ui to tell it to refresh itself when it saves (any time something changes). To do the kicking, I added a @Published kicker property to the class which any views can use to be notified when they need to be torn down and rebuilt. Any time the background context saves, the kicker toggles and that kick is pushed out into the environment.

    Here's the ModelController:

        public class ModelController: ObservableObject {
       
        // MARK: - Properties
        
        let stack: ModelStack
        
        public let viewContext: NSManagedObjectContext
        public let workContext: NSManagedObjectContext
            
        @Published public var uiKicker = true
        
        // MARK: - Public init
        
        public init(stack: ModelStack) {
            
            self.stack = stack
            
            viewContext = stack.persistentContainer.viewContext
            viewContext.automaticallyMergesChangesFromParent = true
            
            workContext = stack.persistentContainer.newBackgroundContext()
            workContext.automaticallyMergesChangesFromParent = true
        }
        
    // Logic logic...
    
        public func save() {
            workContext.performAndWait {
                if workContext.hasChanges {
                    do {
                        try self.workContext.save()
                    } catch {
                        fatalError(error.localizedDescription)
                    }
                }
            }            
            uiKicker.toggle()
        }
    }
    

    I currently instantiate ModelController in @main and inject it into the environment to do my bidding:

    @main
    struct MyApp: App {
        let modelController = ModelController(stack: ModelStack())
        var body: some Scene {
            WindowGroup {
                MainView()
                    .environment(\.managedObjectContext, modelController.viewContext)
                    .environmentObject(modelController)
            }
        }
    }
    

    Now take a view that isn't responding... Here's one now! We can use the uiKicker property to force the stubborn view to refresh. To do that you need to actually use the kicker's value somewhere in your view. It doesn't apparently need to actually change something, just be used - so for example in this view you'll see at the very end I'm setting the opacity of the view based on the uiKicker. It just happens the opacity is set to the same value whether it's true or false so this isn't a noticeable change for the user, other than the fact that the 'sticky' value (in this case list.openItemsCount) gets refreshed.

    You can use the kicker anywhere in the UI and it should work (I've got it on the enclosing VStack but it could be anywhere in there).

    struct CardView: View {
        @ObservedObject var list: Model.List
        @EnvironmentObject var modelController: ModelController
        var body: some View {
            VStack {
                HStack {
                    Image(systemName: "gear")
                    Spacer()
                    Label(String(list.openItemsCount), systemImage: "cart")
                }
                Spacer()
                Text(list.name ?? "Well crap I don't know.")
                Spacer()
                HStack {
                    Image(systemName: "trash")
                        .onTapGesture {
                            modelController.delete(list.objectID)
                        }
                    Spacer()
                    Label("2", systemImage: "person.crop.circle.badge.plus")
                }
            }
            .padding()
            .background(Color.gray)
            .cornerRadius(30)
            .opacity(modelController.uiKicker ? 100 : 100)
        }
    }
    

    And there you have it. Use the uiKicker anywhere things aren't refreshing properly. Literally a few lines of code and stale relationships are a thing of the past!

    As I learn more about SwiftUI I have to say I'm loving it!

    EDIT TO ADD:

    Poking around some more I've found that this only works if the observed object is injected using .environmentObject, it doesn't work if you use custom environment keys and inject using .environment(\.modelController). I have no idea why but it's true as of iOS 14.3/XCode 12.3.

提交回复
热议问题