【问题标题】:SwiftUI 3 MacOs Table single selection and double click open sheetSwiftUI 3 MacOs Table 单选双击打开工作表
【发布时间】:2021-08-30 20:05:42
【问题描述】:
import SwiftUI

struct ContentView: View {
    
    @State private var items: [ItemModel] = Array(0...100).map { ItemModel(id: $0, title: "item \($0)", age: $0) }
    @State private var selection = Set<ItemModel.ID>()
    @State private var sorting = [KeyPathComparator(\ItemModel.age)]
    
    var body: some View {
        Table(items, selection: $selection, sortOrder: $sorting) {
            TableColumn("id", value: \.id) { Text("\($0.id)") }
            TableColumn("title", value: \.title)
            TableColumn("age", value: \.age) { Text("\($0.age)") }
        }
        .onChange(of: sorting) {
            items.sort(using: $0)
        }
        .font(.caption)
        .frame(width: 960, height: 540)
    }
}

struct ItemModel: Identifiable {
    var id: Int
    var title: String
    var age: Int
}

struct ContentView_Previews: PreviewProvider {
    static var previews: some View {
        ContentView()
    }
}

这是一个在 Model.age 上排序的表格的工作示例,并且支持多选, 我想要单选并双击一行打开工作表,这可能吗? 还有如何获取选定的项目对象?

谢谢你????

【问题讨论】:

  • 要进行单选,您只需将选择@State 更改为 ItemModel.ID?而不是相同类型的 Set ,但我也在努力解决其他问题,因为没有 .onTap 或类似的表行处理程序,所以处理点击来做事。

标签: xcode macos swiftui xcode13


【解决方案1】:

您必须将Set&lt;Value.ID&gt; 更改为Value.ID 以仅选择一行,并在文本中设置TapGesture

@State private var selection = Set<ItemModel.ID>() // <-- Use this for multiple rows selections
@State private var selection : ItemModel.ID? // <--- Use this for only one row selection
struct ContentView: View {
    
    @State private var items: [ItemModel] = Array(0...100).map { ItemModel(id: $0, title: "item \($0)", age: $0) }
    //@State private var selection = Set<ItemModel.ID>() <-- Use this for multiple rows selections
    @State private var selection : ItemModel.ID? // <--- Use this for only one row selection
    @State private var sorting = [KeyPathComparator(\ItemModel.age)]
    @State private var showRow = false
    
    var editRow: some View {
        VStack {
            Text(items[selection!].title)
                .font(.title)
            Text("Selected: \(selection.debugDescription)")
             Button("Dismiss") {
                showRow.toggle()
             }.padding()
        }
        .frame(minWidth:400, minHeight: 400)
    }
    
    var body: some View {
        VStack {
            Table(items, selection: $selection, sortOrder: $sorting) {
                TableColumn("id", value: \.id) {
                    Text("\($0.id)")
                        .onTapGesture(count: 2, perform: {
                            if selection != nil {
                                showRow.toggle()
                            }
                        })
                }
                TableColumn("title") { itemModel in
                    Text(itemModel.title)
                        .onTapGesture(count: 2, perform: {
                            if selection != nil {
                                showRow.toggle()
                            }
                        })
                }
                TableColumn("age", value: \.age) { Text("\($0.age)") }
            }
            .onChange(of: sorting) {
                items.sort(using: $0)
            }
            .font(.caption)
            .frame(width: 960, height: 540)
        }
        .sheet(isPresented: $showRow) {
            editRow
        }
    }
}

【讨论】:

  • 很好的答案。我发现我还想要双击更改选择。我没有测试if selection != nil,而是设置了selection = itemModel.id
  • 经过进一步测试,我发现此解决方案存在重大问题。连续单击Text 视图和单击连续空白区域时会有不同的行为。应用于Text 视图的点击手势会覆盖行自身的选择机制。