繁体   English   中英

SwiftUI - 从核心数据 NSSet 中删除 - @FetchRequest with.onDelete()

[英]SwiftUI - Deleting from a Core Data NSSet - @FetchRequest with .onDelete()

我正在努力完成免费和公开的 Hacking With Swift 课程。 不幸的是,Core Data 一章没有解释如何在一对多关系中使用的 NSSet 中删除 object。 我查看了文档并试图弄清楚如何使用 Generated 访问器,但收效甚微。 如果有人能够解释如何做到这一点,那将不胜感激,因为我最有兴趣知道!
一对多关系

到目前为止我尝试过的都失败了!!!


 func deleteCandy (at offsets: IndexSet) {
    for offset in offsets {
        let candy = countries[offset].candyArray[offset]
        moc.delete(candy)
    if self.moc.hasChanges{
        try? moc.save()
        }
    }
}

}

Class:

扩展国家{

@nonobjc public class func fetchRequest() -> NSFetchRequest<Country> {
    return NSFetchRequest<Country>(entityName: "Country")
}

@NSManaged public var fullName: String?
public var wrappedFullName: String {
    fullName ?? "Unknown Full Name"
}
@NSManaged public var shortName: String?
public var wrappedShortName: String{
    shortName ?? "Unknown Short Name"

}
public var candyArray:[Candy] {
    let set = candy as? Set<Candy> ?? []
    return set.sorted {
        $0.wrappedName < $1.wrappedName
    }
}


@NSManaged public var candy: NSSet?

}

// MARK: 为糖果扩展 Country { 生成的访问器

@objc(addCandyObject:)
@NSManaged public func addToCandy(_ value: Candy)

@objc(removeCandyObject:)
@NSManaged public func removeFromCandy(_ value: Candy)

@objc(addCandy:)
@NSManaged public func addToCandy(_ values: NSSet)

@objc(removeCandy:)
@NSManaged public func removeFromCandy(_ values: NSSet)

}

ContentView @Environment(.managedObjectContext) var moc @FetchRequest(entity: Country.entity(), sortDescriptors: []) var countries: FetchedResults

var body: some View {

    VStack {
        List {
            ForEach(countries, id: \.self) { country in
                Section(header: Text(country.wrappedFullName)) {
                    ForEach(country.candyArray, id: \.self) { candy in
                        Text(candy.wrappedName)
                    }
                }
            }.onDelete(perform: deleteCandy)
        }

        Button("Add") {
            let candy1 = Candy(context: self.moc)
            candy1.name = "Mars"
            candy1.origin = Country(context: self.moc)
            candy1.origin?.shortName = "UK"
            candy1.origin?.fullName = "United Kingdom"

            let candy2 = Candy(context: self.moc)
            candy2.name = "KitKat"
            candy2.origin = Country(context: self.moc)
            candy2.origin?.shortName = "UK"
            candy2.origin?.fullName = "United Kingdom"

            let candy3 = Candy(context: self.moc)
            candy3.name = "Twix"
            candy3.origin = Country(context: self.moc)
            candy3.origin?.shortName = "UK"
            candy3.origin?.fullName = "United Kingdom"

            let candy4 = Candy(context: self.moc)
            candy4.name = "Toblerone"
            candy4.origin = Country(context: self.moc)
            candy4.origin?.shortName = "CH"
            candy4.origin?.fullName = "Switzerland"

            try? self.moc.save()
        }
    }

}

func deleteCandy (at offsets: IndexSet) {
    for offset in offsets {
        let candy = countries[offset].candyArray[offset]
        moc.delete(candy)
    if self.moc.hasChanges{
        try? moc.save()
        }
    }
}

用户界面

最后,这个从 DMG 传入闭包的解决方案效果很好:

在这里很好地解释了解决方案。

func deleteCandies(at offsets: IndexSet, from country: Country) {
    for offset in offsets.sorted().reversed() {
        let candyToDelete = country.candyArray[offset]
        country.removeFromCandy(candyToDelete)
        moc.delete(candyToDelete)
    }
    if moc.hasChanges{
        try? moc.save()
    }
}



VStack {
        List {
            ForEach(countries, id: \.self) { country in
                Section(header: Text(country.wrappedFullName)) {
                    ForEach(country.candyArray, id: \.self) { candy in
                        Text(candy.wrappedName)
                    }
                    .onDelete(perform: { offsets in
                        self.deleteCandies(at: offsets, from: country)
                    })
                }
            }
        }

我偶然发现了同样的问题,并使用从 NSManagedObject 子类生成的removeFromCandy(_ value: Candy) function 解决了它。

    var body: some View {

        VStack {
            List {
                ForEach(countries, id: \.self) { country in
                    Section(header: Text(country.wrappedFullName)) {
                        ForEach(country.candyArray, id: \.self) { candy in
                            Text(candy.wrappedName)
                        }.onDelete {indexSet in
                            let deleteItem = self.country.candyArray[indexSet.first!]
                            self.country.removeFromCandy(deleteItem)
                            self.moc.delete(deleteItem)

                            do {
                                try self.moc.save()
                            } catch {
                                print(error)
                            }               
                        }
                    }
                }
            }

            Button("Add") {
                let candy1 = Candy(context: self.moc)
                candy1.name = "Mars"
                candy1.origin = Country(context: self.moc)
                candy1.origin?.shortName = "UK"
                candy1.origin?.fullName = "United Kingdom"

                try? self.moc.save()
            }
        }

    }

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM