使用 NSPasteboard 重新排序和移动 NSTableView 行

问题描述

我有一个 NSTableView,我可以在其中拖放表格行以重新排序它们。这通过在我的视图控制器中设置拖动类型来工作:

@IBOutlet weak var tableView: NSTableView!
let dragType = NSPasteboard.PasteboardType(rawValue: "myapp.task")

override func viewDidLoad() {
  super.viewDidLoad()
  tableView.registerForDraggedTypes([dragType])
}

...然后使用这些表委托方法实现重新排序:

//Start drag
func tableView(_ tableView: NSTableView,pasteboardWriterForRow row: Int) -> NSPasteboardWriting? {
  let item = NSPasteboardItem()
  item.setString(String(row),forType: dragType)
  return item
}

//Verify proposed drop
func tableView(_ tableView: NSTableView,validateDrop info: NSDraggingInfo,proposedRow row: Int,proposedDropOperation dropOperation: NSTableView.DropOperation) -> NSDragOperation {
  if dropOperation == .above {
    return .move
  }else{
    return []
  }
}

//Accept drop of one or multiple rows
func tableView(_ tableView: NSTableView,acceptDrop info: NSDraggingInfo,row: Int,dropOperation: NSTableView.DropOperation) -> Bool {
  var oldindexes = [Int]()
  info.enumerateDraggingItems(options: [],for: tableView,classes: [NSPasteboardItem.self],searchOptions: [:]) { dragItem,_,_ in
    if let str = (dragItem.item as! NSPasteboardItem).string(forType: self.dragType),let index = Int(str) {
      oldindexes.append(index)
    }
  }

  //Do a bunch of logic to reorder the table rows...
}

现在,除了重新排列表格行之外,我还希望能够将一行拖放到我的应用程序中的其他位置——有点像将行移动到不同的位置。

我有一个自定义 NSView 设置为拖动目标,我可以拖动一个表格行,并且自定义视图对拖动一个表格行做出适当的反应:

class MyCustomView: NSView{

  required init?(coder: NSCoder) {
    super.init(coder: coder)
    let taskDragType = NSPasteboard.PasteboardType(rawValue: "myapp.task")
    registerForDraggedTypes([taskDragType])
  }

  override func draggingEntered(_ sender: NSDraggingInfo) -> NSDragOperation {
    //...
  }
  override func draggingExited(_ sender: NSDraggingInfo?) {
    //...
  }
}

但我不清楚的部分是如何在发生放置时获取表行及其关联对象设置为 NSTableCellView 上的属性

//This is another method in MyCustomView

override func performDragOperation(_ draggingInfo: NSDraggingInfo) -> Bool {

  guard let items = draggingInfo.draggingPasteboard.pasteboardItems else{ return false }

  for item in items{
    print("---")
    print(item) //<-- NSPasteboardItem
    let index = item.propertyList(forType: NSPasteboard.PasteboardType(rawValue: "myapp.task"))
    print(index) //<-- Index of the table row
    //How can I also get the task object associated with the row?
  }
}

我可以获得行的索引,但我需要的是来自行数据源的整个对象,以便我可以对它所代表的对象采取行动。我怀疑我需要更改使用 pasteboardWriterForRow 将对象放在粘贴板上的方式,但我不确定如何做。

如何将行索引对象传递给粘贴板?

解决方法

发布这篇文章后不久,我决定尝试一些疯狂的事情,结果发现我找到了一种方法来让这件事变得更简单。 NSPasteboard 似乎只有在您需要将东西放入和取出应用程序时才是真正必要的。由于我只是将某些内容从应用的一个部分移动到另一个部分,因此我可以将拖放委托方法用作事件并自己处理数据。

首先,我设置了一个全局数组,用于添加拖拽的任务对象:

var draggedTasks = [Task]()

每当从我的 NSTableView 中拖动任务时,我都会将它们添加到上述拖动开始的委托方法中的数组中:

//Start drag
func tableView(_ tableView: NSTableView,pasteboardWriterForRow row: Int) -> NSPasteboardWriting? {
  //Queue tasks for moving to phases or projects
  draggedTasks.append(tasks[row])

  //Queue row for reordering
  let item = NSPasteboardItem()
  item.setString(String(row),forType: dragType)
  return item
}

然后我接受 MyCustomView 中的下降,我对 draggedTasks 数组采取行动:

//Save dropped tasks
override func performDragOperation(_ draggingInfo: NSDraggingInfo) -> Bool {
  //Do stuff to draggedTasks based on the context of where they are dropped
  return true
}

这比沿着 NSPasteboard 路线走要简单得多。 ?

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...