一尘不染

SwiftUI拖放文件

swift

我正在尝试向我的SwiftUI Mac应用程序添加“拖放”手势/功能。

我想将文件从系统/桌面拖放到应用程序中。我发现它可以在常规Swift中使用。我现在正尝试在SwiftUI中执行此操作。

onDrop()在SwiftUI for Views中找到了一个函数。但是,看起来这仅适用于应用程序内部的内部手势。我想从外面拖文件。

在Swift中,您需要为拖动的类型注册NSView。

registerForDraggedTypes([kUTTypeFileURL,kUTTypeImage])

我想创建一个NSViewRepresentable并将其包装到我的SwiftUI视图中。

这是我想出的代码,但是我不能调用registerForDraggedTyped

final class DragDropView: NSViewRepresentable {

    func makeNSView(context: NSViewRepresentableContext<DragDropView>) -> NSView {
        let view = NSView()

        view.registerForDraggedTypes([NSPasteboard.PasteboardType.pdf, NSPasteboard.PasteboardType.png])

        return view
    }

    func updateNSView(_ nsView: NSView, context: NSViewRepresentableContext<DragDropView>) {

    }

在SwiftUI中,有没有更简单的解决方案?我想使用该onDrop()函数,但这不适用于外部文件,不是吗?


阅读 787

收藏
2020-07-07

共1个答案

一尘不染

这是拖放的演示,已通过Xcode 11.4 / macOS 10.15.4测试。

Initial image位于资产库上,接受从Finder /
Desktop(拖放)到TextEdit(拖动)作为文件url的拖放(仅出于简化目的),并注册TIFF表示的拖动。

struct TestImageDragDrop: View {
    @State var image = NSImage(named: "image")
    @State private var dragOver = false

    var body: some View {
        Image(nsImage: image ?? NSImage())
            .onDrop(of: ["public.file-url"], isTargeted: $dragOver) { providers -> Bool in
                providers.first?.loadDataRepresentation(forTypeIdentifier: "public.file-url", completionHandler: { (data, error) in
                    if let data = data, let path = NSString(data: data, encoding: 4), let url = URL(string: path as String) {
                        let image = NSImage(contentsOf: url)
                        DispatchQueue.main.async {
                            self.image = image
                        }
                    }
                })
                return true
            }
            .onDrag {
                let data = self.image?.tiffRepresentation
                let provider = NSItemProvider(item: data as NSSecureCoding?, typeIdentifier: kUTTypeTIFF as String)
                provider.previewImageHandler = { (handler, _, _) -> Void in
                    handler?(data as NSSecureCoding?, nil)
                }
                return provider
            }
            .border(dragOver ? Color.red : Color.clear)
    }
}
2020-07-07