.= 'tag.htm'; break; case 'flag': $pre .= $default_pre .= 'flag.htm'; break; case 'my': $pre .= $default_pre .= 'my.htm'; break; case 'my_password': $pre .= $default_pre .= 'my_password.htm'; break; case 'my_bind': $pre .= $default_pre .= 'my_bind.htm'; break; case 'my_avatar': $pre .= $default_pre .= 'my_avatar.htm'; break; case 'home_article': $pre .= $default_pre .= 'home_article.htm'; break; case 'home_comment': $pre .= $default_pre .= 'home_comment.htm'; break; case 'user': $pre .= $default_pre .= 'user.htm'; break; case 'user_login': $pre .= $default_pre .= 'user_login.htm'; break; case 'user_create': $pre .= $default_pre .= 'user_create.htm'; break; case 'user_resetpw': $pre .= $default_pre .= 'user_resetpw.htm'; break; case 'user_resetpw_complete': $pre .= $default_pre .= 'user_resetpw_complete.htm'; break; case 'user_comment': $pre .= $default_pre .= 'user_comment.htm'; break; case 'single_page': $pre .= $default_pre .= 'single_page.htm'; break; case 'search': $pre .= $default_pre .= 'search.htm'; break; case 'operate_sticky': $pre .= $default_pre .= 'operate_sticky.htm'; break; case 'operate_close': $pre .= $default_pre .= 'operate_close.htm'; break; case 'operate_delete': $pre .= $default_pre .= 'operate_delete.htm'; break; case 'operate_move': $pre .= $default_pre .= 'operate_move.htm'; break; case '404': $pre .= $default_pre .= '404.htm'; break; case 'read_404': $pre .= $default_pre .= 'read_404.htm'; break; case 'list_404': $pre .= $default_pre .= 'list_404.htm'; break; default: $pre .= $default_pre .= theme_mode_pre(); break; } if ($config['theme']) { $conffile = APP_PATH . 'view/template/' . $config['theme'] . '/conf.json'; $json = is_file($conffile) ? xn_json_decode(file_get_contents($conffile)) : array(); } !empty($json['installed']) and $path_file = APP_PATH . 'view/template/' . $config['theme'] . '/htm/' . ($id ? $id . '_' : '') . $pre; (empty($path_file) || !is_file($path_file)) and $path_file = APP_PATH . 'view/template/' . $config['theme'] . '/htm/' . $pre; if (!empty($config['theme_child']) && is_array($config['theme_child'])) { foreach ($config['theme_child'] as $theme) { if (empty($theme) || is_array($theme)) continue; $path_file = APP_PATH . 'view/template/' . $theme . '/htm/' . ($id ? $id . '_' : '') . $pre; !is_file($path_file) and $path_file = APP_PATH . 'view/template/' . $theme . '/htm/' . $pre; } } !is_file($path_file) and $path_file = APP_PATH . ($dir ? 'plugin/' . $dir . '/view/htm/' : 'view/htm/') . $default_pre; return $path_file; } function theme_mode_pre($type = 0) { global $config; $mode = $config['setting']['website_mode']; $pre = ''; if (1 == $mode) { $pre .= 2 == $type ? 'portal_category.htm' : 'portal.htm'; } elseif (2 == $mode) { $pre .= 2 == $type ? 'flat_category.htm' : 'flat.htm'; } else { $pre .= 2 == $type ? 'index_category.htm' : 'index.htm'; } return $pre; } ?>List view is not updating when iCloud is activated - Stack Overflow
最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

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

programmeradmin0浏览0评论

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 Improve this question 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)
        }
    }
}
发布评论

评论列表(0)

  1. 暂无评论