带有选择器分段控件子视图和选择手势的列表项

问题描述

我有一个 List 和包含 Picker 视图(分段控制麦粒肿)的项目。我想分别处理选择和选择器状态。未选择列表项时应禁用选择器。

问题是:

  • 第一次点击 - 选择列表项。 (选择 = 真)
  • 点击选择器 - 设置选择 = false

在 UIKit、Button/SegmentControl/等上...抓住“点击”,不要进入 TableView 选择状态。

struct ListView: View {
    @State var selected = Set<Int>()
    let items = (1...10).map { ItemDataModel(id: $0) }
    
    var body: some View {
        List(items) { item in
            ListItemView(dataModel: item)
                .onTapGesture { if !(selected.remove(item.id) != .none) { selected.insert(item.id) }}
        }
    }
}

struct ListItemView: View {
    @Observedobject var dataModel: ItemDataModel
    
    var body: some View {
        let pickerBinding = Binding<Int>(
            get: { dataModel.state?.rawValue ?? -1 },set: { dataModel.state = DataState(rawValue: $0) }
        )
        HStack {
            Text(dataModel.title)
            Spacer()
            Picker("sdf",selection: pickerBinding) {
                Text("State 1").tag(0)
                Text("State 2").tag(1)
                Text("State 3").tag(2)
            }.pickerStyle(SegmentedPickerStyle())
        }
    }
}


enum DataState: Int {
    case state1,state2,state3
}

class ItemDataModel: Hashable,Identifiable,ObservableObject {
    let id: Int
    let title: String
    @Published var state: DataState? = .state1
    
    init(id: Int) {
        self.id = id
        title = "item \(id)"
    }
    
    func hash(into hasher: inout Hasher) {
        hasher.combine(title)
    }
    
    static func == (lhs: ItemDataModel,rhs: ItemDataModel) -> Bool {
        return lhs.id == rhs.id
    }
}

[请尽量忽略语法错误(如果存在)并专注于手势问题]

解决方法

一种可能的解决方案是仅在未选择 Picker 时应用修饰符:

struct ListView: View {
    @State var selected = Set<Int>()
    let items = (1 ... 10).map { ItemDataModel(id: $0) }

    var body: some View {
        List(items) { item in
            if selected.contains(item.id) {
                ListItemView(dataModel: item)
            } else {
                ListItemView(dataModel: item)
                    .disabled(true)
                    .onTapGesture {
                        if !(selected.remove(item.id) != .none) {
                            selected.insert(item.id)
                        }
                    }
            }
        }
    }
}