При сохранении основного объекта данных в popover в SwiftUI возникает ошибка nilError без повторной передачи .environment в SubView - PullRequest
6 голосов
/ 10 октября 2019

Игра с SwiftUI и Core Data привела меня к любопытной проблеме. Таким образом, ситуация следующая:

У меня есть главное представление «AppView» и вспомогательное представление с именем «SubView». Представление SubView будет открыто из представления AppView, если я нажму кнопку «плюс» в NavigationTitleBar как всплывающее окно или лист.

@Environment(\.managedObjectContext) var managedObjectContext
@State private var modal: Bool = false
...
Button(action: {
        self.modal.toggle()
      }) {
        Image(systemName: "plus")
      }.popover(isPresented: self.$modal){
        SubView()
      }

Представление SubView имеет небольшую форму с двумя объектами TextField для добавления имени ифамилия. Входные данные этих двух объектов обрабатываются двумя отдельными свойствами @State. Третий объект в этой форме - простая кнопка, которая должна сохранить имя и фамилию в прикрепленном объекте клиента для CoreData.

...
@Environment(\.managedObjectContext) var managedObjectContext
...
Button(action: {
  let customerItem = Customer(context: self.managedObjectContext)
  customerItem.foreName = self.forename
  customerItem.surname = self.surname

  do {
    try self.managedObjectContext.save()
  } catch {
    print(error)
  }
}) {
  Text("Speichern")
}

Если я пытаюсь сохранить объект клиента таким образом, я получаюerror: "nilError", в частности: "неразрешенная ошибка Error Domain = Foundation._GenericObjCError Code = 0" (null) ", [:]" из NSError.

Но после выяснения, что при добавлении .environment(\.managedObjectContext, context) к вызову SubView () примерно так: SubView().environment(\.managedObjectContext, context) он работает как шарм.

Кто-нибудь знает, почему мне нужно передать managedObjectContext во второй раз? Я подумал, что мне просто нужно передать managedObjectContext один раз, чтобы использовать его во всей иерархии представлений, как в SceneDelegate.swift:

    // Get the managed object context from the shared persistent container.
    let context = (UIApplication.shared.delegate as! AppDelegate).persistentContainer.viewContext

    // Create the SwiftUI view and set the context as the value for the managedObjectContext environment keyPath.
    // Add `@Environment(\.managedObjectContext)` in the views that will need the context.
    let contentView = AppView().environment(\.managedObjectContext, context)

Это потому, что вызов SubView () таким образом,представление не является частью иерархии представлений? Я не понимаю этого ...

1 Ответ

2 голосов
/ 04 ноября 2019

ВАУ ЭТО ПОБЕЛА МЕНЯ ОРЕХАМИ! Тем более, что ошибки не дают вам абсолютно никакой информации о том, как исправить.

Вот исправление, пока ошибка в Xcode не будет устранена:

        .navigationBarItems(trailing:
            Button(action: {
                self.add = true
            }, label: {
                Text("Add Todo List")
            }).sheet(isPresented: $add, content: {
                AddTodoListView().environment(\.managedObjectContext, managedObjectContext)
            })
        )

Просто добавьте .environment(\.managedObjectContext, managedObjectContext) в свой дополнительныйview (модальный, в этом примере).

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...