我正在尝试向我的SwiftUI Mac应用程序添加“拖放”手势/功能。
我想将文件从系统/桌面拖放到应用程序中。我发现它可以在常规Swift中使用。我现在正尝试在SwiftUI中执行此操作。
我onDrop()在SwiftUI for Views中找到了一个函数。但是,看起来这仅适用于应用程序内部的内部手势。我想从外面拖文件。
onDrop()
在Swift中,您需要为拖动的类型注册NSView。
registerForDraggedTypes([kUTTypeFileURL,kUTTypeImage])
我想创建一个NSViewRepresentable并将其包装到我的SwiftUI视图中。
这是我想出的代码,但是我不能调用registerForDraggedTyped。
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()函数,但这不适用于外部文件,不是吗?
这是拖放的演示,已通过Xcode 11.4 / macOS 10.15.4测试。
Initial image位于资产库上,接受从Finder / Desktop(拖放)到TextEdit(拖动)作为文件url的拖放(仅出于简化目的),并注册TIFF表示的拖动。
image
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) } }