首页 > 解决方案 > SwiftUI 列表“NSInternalInconsistencyException”,原因:“尝试插入第 0 节,但更新后只有 0 节”

问题描述

我有以下代码。它需要 Xcode 13 beta 来编译。在 iOS 14 上运行并点击“插入数据”按钮时,它会崩溃并出现以下错误:

libc++abi: terminating with uncaught exception of type NSException
*** Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'attempt to insert section 0 but there are only 0 sections after the update'
terminating with uncaught exception of type NSException
class MyViewModel: ObservableObject {
    @Published var components: [String] = []
}

struct ComponentListContent: View {
    @EnvironmentObject var viewModel: MyViewModel

    var body: some View {
        ForEach(self.$viewModel.components, id: \.self) { $component in // This syntax requires Xcode 13
            TextField("Component Name", text: $component)
        }
    }
}

struct MySheet: View {
    @EnvironmentObject var viewModel: MyViewModel
    @State private var isSheetPresented = true
    @State var isEditing = false

    var body: some View {
        NavigationView {
            List {
                ComponentListContent()
            }
            .environment(\.editMode, .constant(self.isEditing ? EditMode.active : EditMode.inactive))
            .navigationBarItems(trailing:
                Button(action: { self.isEditing.toggle() }, label: {
                    Text(self.isEditing ? "Done" : "Edit")
                })
            )
            .sheet(isPresented: self.$isSheetPresented) {
                Button("Insert data") {
                    self.isSheetPresented.toggle()
                    self.viewModel.components = ["a", "b", "c", "d"]
                }
            }
            .navigationBarTitle("New Analysis", displayMode: .inline)
        }
        .navigationViewStyle(StackNavigationViewStyle())
    }
}

struct ContentView: View {
    var body: some View {
        Text("Hello, world!")
            .sheet(isPresented: .constant(true)) {
                MySheet()
                    .environmentObject(MyViewModel())
            }
    }
}

如果我将主体ComponentListView移入MySheet,则不会发生错误。如果我删除.environment(\.editMode....,它也不会发生。但是我确实需要这两个。是否存在我遗漏的潜在问题?

标签: swiftuixcode13

解决方案


这似乎是由于.constant绑定而发生的。

我建议您切换到 storageEditMode而不是Bool,并isEditing使用将修改它的 getter/setter 进行定义。你可以用 做同样的事情@Binding,但这看起来更干净一些。

@State var editMode = EditMode.inactive
var isEditing: Bool {
    get {
        editMode.isEditing
    }
    nonmutating set {
        if newValue {
            editMode = .active
        } else {
            editMode = .inactive
        }
    }
}

// subscribe
.environment(\.editMode, $editMode)

推荐阅读