swift - 如何获取动态 List / ForEach 可绑定元素的索引(新 Xcode 13 的语法)?
问题描述
到目前为止,要在动态集合中的元素和 List 的行之间设置绑定,我们必须这样做:
List(Array(zip(data.indices, data)), id: \.1.id) { index, _ in
HStack {
Text((index + 1).description)
TextField("", text: Binding(
get: { data[index].text },
set: { data[index].text = $0 }
))
}
}
我们需要: 用于绑定的元素的索引;+ 的元素标识符List
(以避免奇怪的动画);以及Binding
在删除最后一行时避免崩溃的自定义。
这很复杂(我不确定它是否非常有效)。自 WWDC21 以来,我们有了一个新语法(可以反向部署):
List($data) { $item in
HStack {
Text("Index ?")
TextField("", text: $item.text)
}
}
它更干净。
但是,虽然强烈建议使用这种新语法,但如果能够在闭包中访问元素的索引,那就太好了。你知道我们该怎么做吗?
编辑 :
我试过这个(它有效),但我觉得这不是正确的方法:
let d = Binding(get: {
Array(data.enumerated())
}, set: {
data = $0.map {$0.1}
})
List(d, id: \.1.id) { $item in
HStack {
Text("\(item.0 + 1)")
TextField("", text: $item.1.text)
}
}
解决方案
您可以自己构建包装器:
struct ListIndexed<Content: View>: View {
let list: List<Never, Content>
init<Data: MutableCollection&RandomAccessCollection, RowContent: View>(
_ data: Binding<Data>,
@ViewBuilder rowContent: @escaping (Data.Index, Binding<Data.Element>) -> RowContent
) where Content == ForEach<[(Data.Index, Data.Element)], Data.Element.ID, RowContent>,
Data.Element : Identifiable,
Data.Index : Hashable
{
list = List {
ForEach(
Array(zip(data.wrappedValue.indices, data.wrappedValue)),
id: \.1.id
) { i, _ in
rowContent(i, Binding(get: { data.wrappedValue[i] }, set: { data.wrappedValue[i] = $0 }))
}
}
}
var body: some View {
list
}
}
用法:
ListIndexed($items) { i, $item in
HStack {
Text("Index \(i)")
TextField("", text: $item.text)
}
}
推荐阅读
- java - 如何将图像从androids apk包文件夹复制到android设备的另一个文件夹
- python - 追加到上一行
- java - 在两个bean之间注入spring bean-实现相同的接口
- javascript - How to add cursor pointer to Chart.js datalabels plugin
- angular - 如何使用 Router 或 ActivatedRoute 从 url 获取参数
- dji-sdk - 错误:无法 catkin 构建 Onboard-SDK-ROS
- php - symfony4,依赖注入,无默认参数
- excel - VBA:如何将 MSXML2.XMLHTTP60 响应连接到 IHTMLDocument(iframe)
- angular - 如何使用Angular将变量保存在本地文件中
- excel - 单击单元格时激活宏