Однако после удаления элемента из списка привязки TextField, кажется, путаются. В частности, когда я редактирую TextField после удаления, текст обновляется в нескольких полях одновременно. Это также может вызвать следующую ошибку при попытке обновить текстовое поле, которое больше не существует: Swift/ContigiousArrayBuffer.swift:675: Неустранимая ошибка: индекс вне диапазона
< strong>Шаги, чтобы воспроизвести проблему:
- Запустите код, указанный ниже.
- Удалите элемент с надписью «текст 2», нажав кнопку «Удалить» рядом с ним.
- Нажмите на текстовое поле «текст 3» и начните вводить текст.
- Обратите внимание, что текстовые поля «текст 3» и «текст 4» обновляются с одним и тем же текстом.
- Теперь удалите «текст 3».
- Теперь начните вводить текст. в «тексте 4»
- Обратите внимание, что приложение аварийно завершает работу из-за ошибки «индекс вне диапазона».
Код: Выделить всё
struct MyDataItem: Identifiable, Equatable {
var id = UUID().uuidString
var text: String = ""
var placeholder: String = ""
static func == (lhs: MyDataItem, rhs: MyDataItem) -> Bool {
return lhs.id == rhs.id &&
lhs.text == rhs.text &&
lhs.placeholder == rhs.placeholder
}
}
struct BugPrepro: View {
@State var data: [MyDataItem] = [
MyDataItem(placeholder: "text 1"),
MyDataItem(placeholder: "text 2"),
MyDataItem(placeholder: "text 3"),
MyDataItem(placeholder: "text 4")
]
var body: some View {
VStack {
ForEach($data, id: \.id) { $dataItem in
LineItemView(dataItem: $dataItem, onDelete: {
data.removeAll { $0.id == dataItem.id }
})
}
}
}
}
struct LineItemView: View, Equatable {
@Binding var dataItem: MyDataItem
var onDelete: () -> Void
var body: some View {
HStack {
TextField(dataItem.placeholder, text: $dataItem.text)
Button(action: onDelete) {
Text("Delete")
.foregroundColor(.red)
}
}
.padding()
.border(.black)
}
static func == (lhs: LineItemView, rhs: LineItemView) -> Bool {
return lhs.dataItem == rhs.dataItem
}
}
- соответствует Identifying, поэтому SwiftUI должен отслеживать идентичность каждого элемента.
Код: Выделить всё
MyDataItem
- Реализуя Equatable в LineItemView, я пытаюсь предотвращайте ненужные обновления представлений для повышения производительности.
- Несмотря на это, удаление элемента приводит к перекрытию привязок TextField.
Почему привязки TextField путаются после удаления элемента при использовании Equatable в представлении? Как я могу предотвратить ненужную повторную визуализацию представлений, не вызывая этой проблемы?
Подробнее здесь: https://stackoverflow.com/questions/791 ... r-deletion