尝试删除最后一个元素时,带有子项的SwiftUI(2.0)列表崩溃

时间:2020-08-08 07:20:52

标签: ios swift swiftui ios14

我有一个带有子代的SwiftUI列表的以下代码,改编自另一个StackOverflow答案:

SwiftUI 2.0 List with children - how to make the tappable area of the disclosure button cover the whole list item

struct ContentView: View {

    @EnvironmentObject var goodies: GlobalGoodies
    
    func listItem(for item: User) -> some View {
        Text(item.name)
    }
    
    var body: some View {
        NavigationView {
            List {
                ForEach(Array(goodies.users.enumerated()), id: \.1.id) { i, group in
                    DisclosureGroup(isExpanded: $goodies.users[i].isExpanded) {
                        ForEach(group.children ?? []) { item in
                            listItem(for:item)
                        }
                        .opacity(goodies.users[i].opacity)
                    } label: {
                        listItem(for: group)
                        .contentShape(Rectangle())
                        .onTapGesture {
                            withAnimation {
                                goodies.users[i].opacity = goodies.users[i].isExpanded ? 0.0 : 1.0
                                goodies.users[i].isExpanded.toggle()
                            }
                        }
                    }
                }
                .onDelete(perform: delete)
            }
        }
    }

    func delete(at offsets: IndexSet) {
        goodies.users.remove(atOffsets: offsets)
    }
}

GlobalGoodies类和User类的源代码,改编自HackingWithSwift教程示例。

struct User: Identifiable {
    let id = UUID()
    var name: String
    var children: [User]? = nil
    
    var isExpanded = false
    var opacity = 0.0
}

class GlobalGoodies: ObservableObject {
    @Published var users = [
        User(name: "Paul", children: [
            User(name: "Jenny")
        ]),
        User(name: "Taylor", children: [
            User(name: "Tyler"),
            User(name: "Luna")
        ]),
        User(name: "Adele", children: nil)
    ]
}

我能够正常查看这些元素,并且能够删除两个(父)项目而没有任何麻烦。但是,当我尝试删除列表中最后一个剩余的(父)元素时,应用程序崩溃并显示以下消息:

Fatal error: Index out of range: file Swift/ContiguousArrayBuffer.swift, line 444

我知道其他人在使用List时也遇到了相同消息的类似错误,但是仅当使用此特定设置来显示可扩展子元素时,才会发生此错误。我遵循了上面提到的HackingWithSwift教程的代码(这只是一个普通的线性列表),并且可以毫无问题地删除其中的所有元素。

我尝试使用新的List(children:),但是由于某种原因,我无法向其中添加onDelete

任何精确定位此错误位置的帮助都将非常有用,因为我尝试过设置断点等以查看问题出在哪里。该元素名义上被删除,但是崩溃似乎发生在调用body期间或之后。

在此先感谢您的帮助!

1 个答案:

答案 0 :(得分:1)

经过更多的探索,我似乎已经能够自己弄清楚这一点。超出范围的错误发生在以下行:

DisclosureGroup(isExpanded: $goodies.users[i].isExpanded) {

(通过替换.constant(true)可以阻止崩溃,我就能知道了。)

我能够通过使用提供给定索引i的自定义绑定的函数来解决此问题,该函数会在访问索引之前检查索引是否在范围内:

func binding(for index: Int) -> Binding<Bool> {
        Binding<Bool>(
            get: {
                if index > goodies.users.count - 1{
                    return false
                }
                return goodies.users[index].isExpanded
            },
            set: {
                if index > goodies.users.count - 1 {
                    // do nothing
                } else {
                    goodies.users[index].isExpanded = $0
                }
            }
        )
}

受影响的行的用法:

DisclosureGroup(isExpanded: binding(for: i)) {

这解决了崩溃的问题。之前很难弄清楚,因为删除列表项后我没有意识到绑定已被访问。希望这可以帮助处于相同位置的人。