我正在尝试重用View,并将其添加到ContentView上
这是我的孩子视图
struct VStackView: View {
@Binding var spacing: Double
@Binding var alignmentIndex: Int
@Binding var elementsCount: Int
private let alignments: [HorizontalAlignment] = [.leading, .center, .trailing]
var body: some View {
VStack(alignment: self.alignments[alignmentIndex], spacing: CGFloat(spacing)) {
ForEach(0..<elementsCount) {
Text("\($0)th View")
}
}
}
}
这是SuperView
Superview具有诸如Stepper,Slider,Picker之类的控件,用于调整VStack的值(对齐,间距等)
,我想根据该值显示结果。但子视图未更改
struct LayoutView: View {
private let layout: StackLayout
@State private var spacing = 0.0
@State private var alignmentIndex = 0
@State private var alignment: HorizontalAlignment = .leading
@State private var elementsCount: Int = 0
private let alignmentsString = [".leading", ".center", ".trailing"]
private let minValue = 0.0
private let maxValue = 100.0
init(_ layout: StackLayout) {
self.layout = layout
}
var body: some View {
NavigationView {
Form {
Section(header: Text("Controls")) {
VStack(alignment: .leading) {
Text("Spacing: \(Int(spacing))").font(.caption)
HStack {
Text("\(Int(minValue))")
Slider(value: $spacing, in: minValue...maxValue, step: 1)
Text("\(Int(maxValue))")
}
Divider()
Picker("alignment", selection: $alignmentIndex) {
ForEach(0..<self.alignmentsString.count) {
Text("\(self.alignmentsString[$0])")
}
}.pickerStyle(SegmentedPickerStyle())
Divider()
Stepper(value: $elementsCount, in: 0...10) {
Text("Element Count: \(elementsCount)")
}
}
}
VStackView(spacing: $spacing, alignmentIndex: $alignmentIndex, elementsCount: $elementsCount)
}
.navigationBarTitle(Text(layout.rawValue), displayMode: .inline)
}
}
}
我也搜索Google,他们推荐@EnviornmentObject。如果正确,何时使用@Binding属性包装器。
不是两种绑定属性吗?
答案 0 :(得分:1)
简单来说,当您想在两个地方共享数据时,可以使用@Binding。
要在多个视图中共享数据时,请使用@Observable或@environmetobject。
您的VStackView中的ForEach循环会产生问题,因为Swiftui不知道如何唯一地标识每个项目,因此当值更改时也不知道如何更新它们。
像这样附加代码:
ForEach(0..<elementsCount, id: \.self) {
Text("\($0)th View")
}