@Binding属性不是SwiftUI中子视图的刷新视图

时间:2020-01-05 14:35:59

标签: ios swift binding swiftui

我正在尝试重用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属性包装器。

不是两种绑定属性吗?

1 个答案:

答案 0 :(得分:1)

简单来说,当您想在两个地方共享数据时,可以使用@Binding。

要在多个视图中共享数据时,请使用@Observable或@environmetobject。

您的VStackView中的ForEach循环会产生问题,因为Swiftui不知道如何唯一地标识每个项目,因此当值更改时也不知道如何更新它们。

像这样附加代码:

 ForEach(0..<elementsCount,  id: \.self) {
     Text("\($0)th View")
 }