Swift: NSTableView drag to reorder item 如何在拖动过程中显示单元格

Swift: NSTableView drag to reorder item how to display cell during dragging

我正在实施拖放以重新排序单元格,一切正常,但我希望在拖动过程中单元格视图跟随鼠标。

我正在使用 NSTableRowView。我需要让它与 NSTableRowView 一起工作。如果你知道怎么做..

实际结果:

我的实际代码:

    public func tableView(_ tableView: NSTableView, pasteboardWriterForRow row: Int) -> NSPasteboardWriting? {
        let account = viewModel.formattedServices[row]

        let pasteboardItem = NSPasteboardItem()
        pasteboardItem.setString(account.id, forType: NSPasteboard.PasteboardType(rawValue: "mymoney.account"))
        return pasteboardItem
    }

    public func tableView(_ tableView: NSTableView, validateDrop info: NSDraggingInfo, proposedRow row: Int, proposedDropOperation dropOperation: NSTableView.DropOperation) -> NSDragOperation {

        if dropOperation == .above {
            return .move
        } else {
            return []
        }
    }

    public func tableView(_ tableView: NSTableView, acceptDrop info: NSDraggingInfo, row: Int, dropOperation: NSTableView.DropOperation) -> Bool {
        guard let item = info.draggingPasteboard.pasteboardItems?.first,
              let theString = item.string(forType: NSPasteboard.PasteboardType(rawValue: "mymoney.account")),
              let account = viewModel.formattedServices.first(where: { [=11=].id == theString }),
              let originalRow = viewModel.formattedServices.firstIndex(where: { [=11=].id == account.id })
                else { return false }

        var newRow = row
        // When you drag an item downwards, the "new row" index is actually --1. Remember dragging operation is `.above`.
        if originalRow < newRow {
            newRow = row - 1
        }

        // Animate the rows
        tableView.beginUpdates()
        tableView.moveRow(at: originalRow, to: newRow)
        tableView.endUpdates()

        return true
    }

我找不到在拖动过程中 return 单元格视图的方法。

对于 return 单元格视图,我使用 func tableView(_ tableView: NSTableView, rowViewForRow row: Int) -> NSTableRowView?

还有我的NSTableRowView

class ServicesTableViewCell: NSTableRowView, CustomCelleable {

    @IBOutlet weak var titleLbl: NSTextField!
    @IBOutlet weak var notificationCount: NSButton!
    @IBOutlet weak var iconImageView: NSImageView!


    override func draw(_ dirtyRect: NSRect) {
        super.draw(dirtyRect)
        isEmphasized = false
    }

    func setup(with item: Celleable) { }

步骤 1 NSTableRowView 无法替换 NSTableCellView,对 table 个单元格使用 NSTableCellView

从测试应用开始。在 IB 中,select table 视图中的列并从库中添加文本 Table 单元格视图。

在 ViewController.swift 中将 tableView(_:rowViewForRow:) 替换为 tableView(_:viewFor:row:)

func tableView(_ tableView: NSTableView, viewFor tableColumn: NSTableColumn?, row: Int) -> NSView? {
    guard let cell = tableView.makeView(withIdentifier: tableColumn!.identifier, owner: self) as? NSTableCellView else {
        fatalError()
    }

    cell.textField?.stringValue = array[row].name

    return cell
}

setTableView() 中删除注册笔尖。

func setTableView() {
    tableView.delegate = self
    tableView.dataSource = self
    tableView.registerForDraggedTypes([NSPasteboard.PasteboardType(rawValue: "mymoney.account")])
}

服务Table不再使用ViewCell。

步骤 2 使用 NSTableRowView 的子 class 禁用 isEmphasized:

添加 NSTableRowView 的子class,我们将其命名为 ServicesTableRowView.

覆盖 isEmphasized 以使其始终 false

class ServicesTableRowView: NSTableRowView {

    override var isEmphasized: Bool {
        get {
            return false
        }
        set {
            // do nothing
        }
    }

}

在 IB select 中 table 视图并从库中添加自定义视图。将视图的 class 设置为 ServicesTableRowView 并将标识符设置为 NSTableViewRowViewKey.

修改后的测试项目:we.tl/t-yjpZrz6q7g