当 SwiftUI Picker 选择更改 EnvironmentObject 时,有没有办法调用函数?

mig*_*now 1 slider picker ios swift swiftui

我有一个正在使用的 SwiftUI 表单,它更新 EnvironmentObject 的值。当 Picker 的值发生变化时,我需要调用一个函数,但我发现的每一种方式要么非常混乱,要么会导致其他问题。我正在寻找的是 Slider 工作的类似方式,但似乎没有。没有任何解决方案的基本代码在这里:

class ValueHolder : ObservableObject {

@Published var sliderValue : Float = 0.5
static let segmentedControlValues : [String] = ["one", "two", "three", "four", "five"]
@Published var segmentedControlValue : Int = 3

}
struct ContentView: View {

    @EnvironmentObject var valueHolder : ValueHolder

    func sliderFunction() {
        print(self.valueHolder.sliderValue)
    }
    func segmentedControlFunction() {
        print(ValueHolder.segmentedControlValues[self.valueHolder.segmentedControlValue])
    }

    var body: some View {
        Form {
            Text("\(self.valueHolder.sliderValue)")
            Slider(value: self.$valueHolder.sliderValue, onEditingChanged: {_ in self.sliderFunction()
            })
            Text("\(ValueHolder.segmentedControlValues[self.valueHolder.segmentedControlValue])")
            Picker("", selection: self.$valueHolder.segmentedControlValue) {
                ForEach(0..<ValueHolder.segmentedControlValues.count) {
                    Text("\(ValueHolder.segmentedControlValues[$0])")
                }
            }.pickerStyle(SegmentedPickerStyle())
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

在这里查看了这个类似(但不同)的问题后:Is there a way to call a function when a SwiftUI Picker selection changes? 我已经尝试使用 onReceive() 如下,但是当 Slider 值更改时也会调用它,从而导致不需要的行为。

.onReceive([self.valueHolder].publisher.first(), perform: {_ in
                self.segmentedControlFunction()
            })
Run Code Online (Sandbox Code Playgroud)

我尝试更改 onReceive 的参数以仅按该值对其进行过滤。传递的值是正确的,但在滑块移动时仍会调用 segmentedControlFunction,而不仅仅是在选择器更改时。

.onReceive([self.valueHolder.segmentedControlValue].publisher.first(), perform: {_ in
                self.segmentedControlFunction()
            })
Run Code Online (Sandbox Code Playgroud)

如何以与sliderFunction 类似的方式调用segmentedControlFunction?

Asp*_*eri 5

有更简单的方法,它看起来更适合我。

通用架构如下

Picker("Label", selection: Binding(    // proxy binding
    get: { self.viewModel.value },     // get value from storage
    set: {
            self.viewModel.value = $0  // set value to storage

            self.anySideEffectFunction() // didSet function
    }) {
       // picker content
    }
Run Code Online (Sandbox Code Playgroud)