ForEach не работает с идентифицируемым & id = UUID()
import SwiftUI
struct TestStudentView: View {
@StateObject var students = Students()
@State private var name = ""
@State private var numberOfSubjects = ""
@State private var subjects = [Subjects](repeating: Subjects(name: "", grade: ""), count: 10)
var body: some View {
NavigationView {
Group {
Form {
Section(header: Text("Student details")) {
TextField("Name", text: $name)
TextField("Number of subjects", text: $numberOfSubjects)
}
let count = Int(numberOfSubjects) ?? 0
Text("Count: \(count)")
Section(header: Text("Subject grades")) {
if count>0 && count<10 {
ForEach(0 ..< count, id: \.self) { number in
TextField("Subjects", text: $subjects[number].name)
TextField("Grade", text: $subjects[number].grade)
}
}
}
}
VStack {
ForEach(students.details) { student in
Text(student.name)
ForEach(student.subjects) { subject in //Does not work as expected
//ForEach(student.subjects, id:\.id) { subject in //Does not work as expected
//ForEach(student.subjects, id:\.self) { subject in //works fine with this
HStack {
Text("Subject: \(subject.name)")
Text("Grade: \(subject.grade)")
}
}
}
}
}
.navigationTitle("Student grades")
.navigationBarItems(trailing:
Button(action: {
let details = Details(name: name, subjects: subjects)
students.details.append(details)
}, label: {
Text("Save")
})
)
}
}
}
struct TestStudentView_Previews: PreviewProvider {
static var previews: some View {
TestStudentView()
}
}
class Students: ObservableObject {
@Published var details = [Details]()
}
struct Details: Identifiable {
let id = UUID()
var name: String
var subjects: [Subjects]
}
struct Subjects: Identifiable, Hashable {
let id = UUID()
var name: String
var grade: String
}
Когда я использую - "ForEach (student.subjects, id: .id) {subject in", при нормальных обстоятельствах предполагается, что он работает как id = UUID, и неправильный вывод выглядит следующим образом:
затем, поскольку класс соответствует Идентифицируемому, я попробовал - "ForEach(student.subjects) {subject in" он все еще работает неправильно. Однако, когда я это сделаю - «ForEach(student.subjects, id:.self) {subject in», за исключением того, что мне нужно было, чтобы класс соответствовал хэшируемому, и дал мне правильный ожидаемый результат. Правильный вывод, который показан:
1 ответ
Вам нужно использовать карту вместо повторения.
Использование Array.init(повторение :) вызовет инициализацию субъектов только один раз, а затем многократно вставит этот объект в массив.
Таким образом, в данном случае все идентификаторы одинаковы.
Вы можете проверить, просто распечатав все идентификаторы этим
.onAppear() { print(subjects.map({ (sub) in print(sub.id) }))
struct TestStudentView: View {
@StateObject var students = Students()
@State private var name = ""
@State private var numberOfSubjects = ""
@State private var subjects: [Subjects] = (0...10).map { _ in
Subjects(name: "", grade: "")
} //<-- Here