带有 onTapGesture 的分段选取器不响应点击

Mic*_*mon 6 segmentedcontrol swiftui

我尝试重新实现我正在使用的 SegmentedControlers,因为它们在 Xcode 11 beta 5 中已被弃用。这花了一段时间但我得到了我想要的外观。但是,当我用 onTapGesture() 替换 tapAction 时,选择器停止工作。

下面的代码显示了问题。注释掉 pickerStyle 会得到一个与 onTapGesture() 一起工作的轮选择器

import SwiftUI

var oneSelected = false
struct ContentView: View {
    @State var sel = 0
    var body: some View {
        VStack {
            Picker("Test", selection: $sel) {
                Text("A").tag(0)
                Text("B").tag(1)
                Text("C").tag(2)
            }
            .pickerStyle(SegmentedPickerStyle())
            Picker("Test", selection: $sel) {
                Text("A").tag(0)
                Text("B").tag(1)
                Text("C").tag(2)
            }
            .pickerStyle(SegmentedPickerStyle())
            .onTapGesture(perform: {
                oneSelected = (self.sel == 1)
            })
            Text("Selected: \(sel)")
        }
    }
}

#if DEBUG
struct ContentView_Previews: PreviewProvider {
    static var previews: some View {
        ContentView()
    }
}
#endif
Run Code Online (Sandbox Code Playgroud)

我希望 Picker().pickerStyle(SegmentedPickerStyle()) 应该像 SegmentedController() 那样工作。

RPa*_*l99 6

您添加的内容tapGesture会干扰选择器的内置点击手势识别,这就是为什么您的代码在.onTapGesture点击选择器时运行,但选择器本身不响应点击。对于您的情况,我建议采用不同的方法:传递一个符合ObservableObject您的视图模型ContentView,并让它包含一个@Published用于选择器选择的变量。然后向该变量添加一个属性观察器,检查所选选项是否为 1。

例如:

class ViewModel: ObservableObject {
    @Published var sel = 0 {
        didSet {
            oneSelected = oldValue == 1
        }
    }
    var oneSelected = false
}
Run Code Online (Sandbox Code Playgroud)

SceneDelegate.swift, 或任何ContentView声明的地方:

ContentView().environmentObject(ViewModel())
Run Code Online (Sandbox Code Playgroud)

ContentView.swift

@EnvironmentObject var env: ViewModel
var body: some View {
    VStack {
        Picker("Test", selection: $env.sel) {
            Text("A").tag(0)
            Text("B").tag(1)
            Text("C").tag(2)
        }
        .pickerStyle(SegmentedPickerStyle())
        Picker("Test", selection: $env.sel) {
            Text("A").tag(0)
            Text("B").tag(1)
            Text("C").tag(2)
        }
        .pickerStyle(SegmentedPickerStyle())
        Text("Selected: \(sel)")
    }
}
Run Code Online (Sandbox Code Playgroud)

注意:根据我的经验,在以前的测试版中添加 TapGestureSegmentedControl会导致SegmentedControl无响应,所以我不确定为什么它在以前的版本中对您有用。从 SwiftUI beta 5 开始,我认为没有办法为手势分配优先级。

编辑:您可以使用.highPriorityGesture()使您的手势优先于视图中定义的手势,但您的手势具有更高的优先级会导致您的问题。但是,您可以使用.simultaneousGesture(),我认为这可以解决您的问题,但我认为它在 SwiftUI Beta 5 中功能不全。