尝试删除最后一个元素时,带有子项的 SwiftUI (2.0) 列表崩溃
SwiftUI (2.0) list with children crashes when trying to delete the last element
我有以下用于包含子项的 SwiftUI 列表的代码,改编自另一个 Whosebug 答案:
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 class 和用户 class 的源代码,改编自 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
.
期间或之后
在此先感谢您的帮助!
经过更多的探索,我似乎已经能够自己解决这个问题。此行发生 out-of-range 错误:
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 = [=11=]
}
}
)
}
受影响行的用法:
DisclosureGroup(isExpanded: binding(for: i)) {
这解决了崩溃问题。之前有点难以确定,因为我没有意识到删除列表项后访问了绑定。希望这对处于相同位置的人有所帮助。
我有以下用于包含子项的 SwiftUI 列表的代码,改编自另一个 Whosebug 答案:
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 class 和用户 class 的源代码,改编自 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
.
在此先感谢您的帮助!
经过更多的探索,我似乎已经能够自己解决这个问题。此行发生 out-of-range 错误:
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 = [=11=]
}
}
)
}
受影响行的用法:
DisclosureGroup(isExpanded: binding(for: i)) {
这解决了崩溃问题。之前有点难以确定,因为我没有意识到删除列表项后访问了绑定。希望这对处于相同位置的人有所帮助。