У меня есть приложение SwiftUI с жизненным циклом приложения SwiftUI, которое включает тип master-detail. список, полученный из CoreData. У меня есть стандартный список в ContentView и NavigationLinks в Детальный вид. Я передаю объект сущности Core Data в Detailview.
Моя проблема заключается в настройке привязок к TextFields в DetailView для ввода данных. и для редактирования. Я попытался создать инициализатор, который не смог заставить работать. У меня есть удалось заставить его работать только со следующим. Присвоение начальных значений внутри тела не кажется лучшим способом сделать это, хотя это работает.
Поскольку объекты Core Data являются ObservableObjects, я подумал, что должен иметь возможность прямой доступ и обновление связанных переменных, но я не смог найти способ сослаться привязка к Core Data в цикле ForEach.
Есть ли способ сделать это более подходящим, чем мой код ниже?
Упрощенный пример:
struct DetailView: View {
var thing: Thing
var count: Int
@State var localName: String = ""
@State private var localComment: String = ""
@State private var localDate: Date = Date()
//this does not work - cannot assign String? to State<String>
// init(t: Thing) {
// self._localName = t.name
// self._localComment = t.comment
// self._localDate = Date()
// }
var body: some View {
//this is the question - is this safe?
DispatchQueue.main.async {
self.localName = self.thing.name ?? "no name"
self.localComment = self.thing.comment ?? "No Comment"
self.localDate = self.thing.date ?? Date()
}
return VStack {
Text("\(thing.count)")
.font(.title)
Text(thing.name ?? "no what?")
TextField("name", text: $localName)
Text(thing.comment ?? "no comment?")
TextField("comment", text: $localComment)
Text("\(thing.date ?? Date())")
//TextField("date", text: $localDate)
}.padding()
}
}
И для полноты ContentView:
struct ContentView: View {
@Environment(\.managedObjectContext) private var viewContext
@FetchRequest(sortDescriptors: [NSSortDescriptor(keyPath: \Thing.date, ascending: false)])
private var things : FetchedResults<Thing>
@State private var count: Int = 0
@State private var coverDeletedDetail = false
var body: some View {
NavigationView {
List {
ForEach(things) { thing in
NavigationLink(destination: DetailView(thing: thing, count: self.count + 1)) {
HStack {
Image(systemName: "gear")
.resizable()
.frame(width: 40, height: 40)
.onTapGesture(count: 1, perform: {
updateThing(thing)
})
Text(thing.name ?? "untitled")
Text("\(thing.count)")
}
}
}
.onDelete(perform: deleteThings)
if UIDevice.current.userInterfaceIdiom == .pad {
NavigationLink(destination: WelcomeView(), isActive: self.$coverDeletedDetail) {
Text("")
}
}
}
.navigationTitle("Thing List")
.navigationBarItems(trailing: Button("Add Task") {
addThing()
})
}
}
private func updateThing(_ thing: FetchedResults<Thing>.Element) {
withAnimation {
thing.name = "Updated Name"
thing.comment = "Updated Comment"
saveContext()
}
}
private func deleteThings(offsets: IndexSet) {
withAnimation {
offsets.map { things[$0] }.forEach(viewContext.delete)
saveContext()
self.coverDeletedDetail = true
}
}
private func addThing() {
withAnimation {
let newThing = Thing(context: viewContext)
newThing.name = "New Thing"
newThing.comment = "New Comment"
newThing.date = Date()
newThing.count = Int64(self.count + 1)
self.count = self.count + 1
saveContext()
}
}
func saveContext() {
do {
try viewContext.save()
} catch {
print(error)
}
}
}
И основные данные:
extension Thing {
@nonobjc public class func fetchRequest() -> NSFetchRequest<Thing> {
return NSFetchRequest<Thing>(entityName: "Thing")
}
@NSManaged public var comment: String?
@NSManaged public var count: Int64
@NSManaged public var date: Date?
@NSManaged public var name: String?
}
extension Thing : Identifiable {
}
Любое руководство будет оценено. Xcode 12.2 iOS 14.2





Вы уже упоминали об этом. CoreData отлично работает со SwiftUI.
Просто сделайте свою вещь как ObservableObject
@ObservedObject var thing: Thing
а затем вы можете передавать значения из вещи как Binding. Это работает и в ForEach.
TextField("name", text: $thing.localName)
Для других — обратите внимание, что мне пришлось использовать расширение Binding выше, поскольку NSManagedObjects являются необязательными. Таким образом, как сказал Давидев:
TextField("name", text: Binding($thing.name, "no name"))
И ObservedObject, а не Observable
Вы совершенно правы. Исправил мой ответ. Спасибо :)
Спасибо. Клянусь, я пробовал это. Должно быть, я ошибся с синтаксисом. Я попробую еще раз.