1

我正在尝试实现NSCollectionView 拖动项目(不仅仅是在其上放置东西)。

在我的示例代码中,我正在通过拖动注册 CollectionView:

collectionView.registerForDraggedTypes([.URL])
collectionView.setDraggingSourceOperationMask(.every, forLocal: false)
collectionView.setDraggingSourceOperationMask(.every, forLocal: true)

然后,我从 NSCollectionViewDelegate 协议中实现了这些方法:

func collectionView(_ collectionView: NSCollectionView, canDragItemsAt indexPaths: Set<IndexPath>, with event: NSEvent) -> Bool {
    return true
}

func collectionView(_ collectionView: NSCollectionView, pasteboardWriterForItemAt indexPath: IndexPath) -> NSPasteboardWriting? {    
    return URL(fileURLWithPath: #file) as NSPasteboardWriting
}

func collectionView(_ collectionView: NSCollectionView, draggingSession session: NSDraggingSession, willBeginAt screenPoint: NSPoint, forItemsAt indexPaths: Set<IndexPath>) { }

func collectionView(_ collectionView: NSCollectionView, draggingSession session: NSDraggingSession, endedAt screenPoint: NSPoint, dragOperation operation: NSDragOperation) { }

但他们甚至都没有被调用过!为什么不?

如果我添加这两种方法:

func collectionView(_ collectionView: NSCollectionView, validateDrop draggingInfo: NSDraggingInfo, proposedIndexPath proposedDropIndexPath: AutoreleasingUnsafeMutablePointer<NSIndexPath>, dropOperation proposedDropOperation: UnsafeMutablePointer<NSCollectionView.DropOperation>) -> NSDragOperation {
    return .move
}

func collectionView(_ collectionView: NSCollectionView, acceptDrop draggingInfo: NSDraggingInfo, indexPath: IndexPath, dropOperation: NSCollectionView.DropOperation) -> Bool {
    return true
}

然后我可以成功地将文件从桌面拖放到集合视图中,但反过来仍然不行。

这是怎么回事?

最好的问候,V。

4

1 回答 1

0

这里完全一样。

看起来像一个错误(Xcode 9.4.1,Swit 4.1.2)。

就像您提到的那样,validateDropacceptDrop在我的应用程序中被调用。
但是(例如),endedAt不:

func collectionView(_ collectionView: NSCollectionView,
                    draggingSession session: NSDraggingSession,
                    endedAt screenPoint: NSPoint,
                    dragOperation operation: NSDragOperation) { }

我能找到的唯一解决方法(对于上述endedAt委托)是子类化NSCollectionView(我ImageCollectionView在下面的示例中调用它)并实现我自己的delegate/protocol

import Cocoa

protocol ImageCollectionViewDelegate {
    func didExitDragging()
}

class ImageCollectionView: NSCollectionView {
    var imageCollectionViewDelegate: ImageCollectionViewDelegate?
}

extension ImageCollectionView {
    override func draggingExited(_ sender: NSDraggingInfo?) {
        super.draggingExited(sender)
        imageCollectionViewDelegate?.didExitDragging()
    }
}

(我不得不称它为imageCollectionViewDelegate不与现有delegate财产冲突。)

然后,在我的控制器中(我正在调用它ImageCollectionViewController):

@IBOutlet internal weak var imageCollectionView: ImageCollectionView! {
    didSet {
        imageCollectionView.imageCollectionViewDelegate = self
    }
}

extension ImageCollectionViewController: ImageCollectionViewDelegate {
    func didExitDragging() {
        highlightDestination(false)
    }
}

这允许我在拖动到集合之外时做一些事情。
在这个非常简单的用例中,只需打开/关闭目标视图的突出显示:

突出显示

理想情况下,不需要所有这些额外的代码。

我也在寻找处理这个问题的正确方法。

于 2018-08-24T10:03:03.983 回答