List view is not updating when iCloud is activated - Stack Overflow

admin2025-04-18  2

The following code is working perfectly fine when iCloud is deactivated. A meeting object is saved to the DB and the list on ContentView is updated properly.

But when I activate iCloud the list is not longer updated.

I tried already to wrap the meeting object with @State and loaded/updated the @Bindable property .onAppear/.onDisappear. But without any effect.

import SwiftData
import SwiftUI

@main
struct SwiftDataRelationNavigationApp: App {
    
    var body: some Scene {
        WindowGroup {
            ContentView()
                .modelContainer(for: Meeting.self)
        }
    }
}
import SwiftData
import SwiftUI
import UniformTypeIdentifiers

@Model
class Meeting {
    var id: UUID = UUID()
    var name: String = ""
    
    @Relationship(deleteRule: .cascade) var topics: [Topic] = [Topic]()
    
    init(id: UUID = UUID(), name: String = "", topics: [Topic] = [Topic]()) {
        self.id = id
        self.name = name
        self.topics = topics
    }
}

@Model
class Topic {
    var id: UUID = UUID()
    var name: String = ""
    
    init(id: UUID = UUID(), name: String = "") {
        self.id = id
        self.name = name
    }
}

struct ContentView: View {
    @Environment(\.modelContext) var modelContext
    @Query var meetings: [Meeting]
    
    @State private var path = NavigationPath()
    
    var body: some View {
        NavigationStack(path: $path) {
            
            VStack(alignment: .leading, spacing: 5) {
                
                //BUG: the list is not updated after I created a new meeting object when iCloud is activated.
                List {
                    ForEach(meetings) { meeting in
                        NavigationLink(meeting.name, value: meeting)
                    }
                }
            }
            .navigationDestination(for: Meeting.self, destination: MeetingDetailView.init)
            
            .toolbar {
                Button("add", action: createMeeting)
            }
        }
        .padding()
    }
    
    func createMeeting() {
        let m = Meeting()
        m.name = "New Meeting"
        
        modelContext.insert(m)
        path.append(m)
    }
}

struct MeetingDetailView: View {
    @Bindable var meeting: Meeting
    
    var body: some View {
        VStack(alignment: .leading, spacing: 5) {
            
            TextField("", text: $meeting.name)
        }
    }
}

The following code is working perfectly fine when iCloud is deactivated. A meeting object is saved to the DB and the list on ContentView is updated properly.

But when I activate iCloud the list is not longer updated.

I tried already to wrap the meeting object with @State and loaded/updated the @Bindable property .onAppear/.onDisappear. But without any effect.

import SwiftData
import SwiftUI

@main
struct SwiftDataRelationNavigationApp: App {
    
    var body: some Scene {
        WindowGroup {
            ContentView()
                .modelContainer(for: Meeting.self)
        }
    }
}
import SwiftData
import SwiftUI
import UniformTypeIdentifiers

@Model
class Meeting {
    var id: UUID = UUID()
    var name: String = ""
    
    @Relationship(deleteRule: .cascade) var topics: [Topic] = [Topic]()
    
    init(id: UUID = UUID(), name: String = "", topics: [Topic] = [Topic]()) {
        self.id = id
        self.name = name
        self.topics = topics
    }
}

@Model
class Topic {
    var id: UUID = UUID()
    var name: String = ""
    
    init(id: UUID = UUID(), name: String = "") {
        self.id = id
        self.name = name
    }
}

struct ContentView: View {
    @Environment(\.modelContext) var modelContext
    @Query var meetings: [Meeting]
    
    @State private var path = NavigationPath()
    
    var body: some View {
        NavigationStack(path: $path) {
            
            VStack(alignment: .leading, spacing: 5) {
                
                //BUG: the list is not updated after I created a new meeting object when iCloud is activated.
                List {
                    ForEach(meetings) { meeting in
                        NavigationLink(meeting.name, value: meeting)
                    }
                }
            }
            .navigationDestination(for: Meeting.self, destination: MeetingDetailView.init)
            
            .toolbar {
                Button("add", action: createMeeting)
            }
        }
        .padding()
    }
    
    func createMeeting() {
        let m = Meeting()
        m.name = "New Meeting"
        
        modelContext.insert(m)
        path.append(m)
    }
}

struct MeetingDetailView: View {
    @Bindable var meeting: Meeting
    
    var body: some View {
        VStack(alignment: .leading, spacing: 5) {
            
            TextField("", text: $meeting.name)
        }
    }
}
Share asked Jan 30 at 12:48 MatFetschMatFetsch 651 silver badge7 bronze badges 9
  • Any change if you call save() directly after doing the insert? – Joakim Danielson Commented Jan 30 at 14:24
  • No. The list view is still not updated from the created object. – MatFetsch Commented Jan 30 at 17:41
  • XCode shows an error which I’m not sure I do understand. Unresolved error loading container Error Domain=NSCocoaErrorDomain Code=134060 "A Core Data error occurred." UserInfo={NSLocalizedFailureReason=CloudKit integration requires that all relationships have an inverse, the following do not: Meeting: topics CloudKit integration requires that all relationships be optional, the following are not: Meeting: topics} – MatFetsch Commented Jan 30 at 17:45
  • Actually the error message says exactly what is wrong, you need to make your relationship properties optional. This is an (unfortunate) requirement when using iCloud together with SwiftData – Joakim Danielson Commented Jan 30 at 19:07
  • 1 Now I got it working. The @Relationship of the meeting object needs the inverse reference to an optional value of meeting at the topic object. @Relationship(deleteRule: .cascade, inverse: \Topic.meeting) var topics: [Topic]? And the topic needs the optional reference to the related meeting. class Topic { var id: UUID = UUID() var name: String = "" var meeting: Meeting? init(id: UUID = UUID(), name: String = "", meeting: Meeting) { self.id = id self.name = name self.meeting = meeting } } – MatFetsch Commented Jan 31 at 6:35
 |  Show 4 more comments

1 Answer 1

Reset to default 0

The following code is working including iCloud synchronization and the changes I made are commented.

The App file stays unchanged.

import SwiftData
import SwiftUI
import UniformTypeIdentifiers

@Model
class Meeting {
    var id: UUID = UUID()
    var name: String = ""
    
    //the relationship is opional now and an inverse target has been added.
    @Relationship(deleteRule: .cascade, inverse: \Topic.meeting) var topics: [Topic]?
    
    init(id: UUID = UUID(), name: String = "") {
        self.id = id
        self.name = name
    }
}

@Model
class Topic {
    var id: UUID = UUID()
    var name: String = ""
    //an optionl reference has been added for the relation with a meeting object.
    var meeting: Meeting?
    
    init(id: UUID = UUID(), name: String = "", meeting: Meeting) {
        self.id = id
        self.name = name
        self.meeting = meeting
    }
}

struct ContentView: View {
    @Environment(\.modelContext) var modelContext
    @Query var meetings: [Meeting]
    
    @State private var path = NavigationPath()
    
    var body: some View {
        NavigationStack(path: $path) {
            
            VStack(alignment: .leading, spacing: 5) {
                
                List {
                    ForEach(meetings) { meeting in
                        NavigationLink(meeting.name, value: meeting)
                    }
                }
            }
            .navigationDestination(for: Meeting.self, destination: MeetingDetailView.init)
            
            .toolbar {
                Button("add", action: createMeeting)
            }
        }
        .padding()
    }
    
    func createMeeting() {
        let m = Meeting()
        m.name = "New Meeting"
        
        do {
            modelContext.insert(m)
            try modelContext.save()
            path.append(m)
        } catch {
            print(error.localizedDescription)
        }
        
    }
}

struct MeetingDetailView: View {
    @Bindable var meeting: Meeting
    
    var body: some View {
        VStack(alignment: .leading, spacing: 5) {
            
            TextField("", text: $meeting.name)
        }
    }
}
转载请注明原文地址:http://anycun.com/QandA/1744918302a89467.html