有没有人能够限制在 SwiftUI 3 中 mac 上放置的对象的类型?

is anyone able to restrict the type of the objects dropped on the mac in SwiftUI 3?

根据文档,它应该非常简单。 List 的示例:https://developer.apple.com/documentation/swiftui/list/ondrop(of:istargeted:perform:)-75hvy#

UTType 应该是限制 SwiftUI 对象可以接收的参数。就我而言,我只想接受 AppsUTType.applicationBundlehttps://developer.apple.com/documentation/uniformtypeidentifiers/uttype/3551459-applicationbundle

但它不起作用。 SwiftUI 对象永远不会改变状态,也永远不会接受放置。闭包永远不会 运行。无论是 ListsH/VStacksButtons 还是什么。 pdf 类型似乎也不起作用,许多其他类型也是如此。如果 fileURL,我唯一可以使用的类型,主要是没有限制。

我不确定我是否做错了什么,或者 SwiftUI 是否为 mac.

工作了一半

代码如下:

List(appsToIgnore, id: \.self, selection: $selection) {
    Text([=11=])
}
.onDrop(of: [.applicationBundle, .application], isTargeted: isTargeted) { providers in
    print("hehe")
    
    return true
}

UTType 数组中替换或仅添加 .fileURL 可以使放置工作但没有任何类型限制。

我也曾尝试在 ForEach 上使用 .onInsert 而不是 (https://developer.apple.com/documentation/swiftui/foreach/oninsert(of:perform:)-2whxl#), and to go through a proper DropDelegate (https://developer.apple.com/documentation/swiftui/dropdelegate#),但一直得到相同的结果。 macOS 的 SwiftUI drop 似乎还没有工作,但我找不到关于此的任何官方信息。在文档中它被写成 macOS 11.0+ 所以我希望它能工作?

感谢任何信息!谢谢。

需要手动验证,使用DropDelegate拖过来的是什么类型的文件。

这是可能方法的简化演示。使用 Xcode 13 / macOS 11.6

进行测试
let delegate = MyDelegate()

...


List(appsToIgnore, id: \.self, selection: $selection) {
    Text([=10=])
}
.onDrop(of: [.fileURL], delegate: delegate) // << accept file URLs

和验证部分如

class MyDelegate: DropDelegate {

    func validateDrop(info: DropInfo) -> Bool {
        // find provider with file URL
        guard info.hasItemsConforming(to: [.fileURL]) else { return false }
        guard let provider = info.itemProviders(for: [.fileURL]).first else { return false }

        var result = false
        if provider.canLoadObject(ofClass: String.self) {
            let group = DispatchGroup()
            group.enter()     // << make decoding sync

            // decode URL from item provider
            _ = provider.loadObject(ofClass: String.self) { value, _ in
                defer { group.leave() }
                guard let fileURL = value, let url = URL(string: fileURL) else { return }

                // verify type of content by URL
                let flag = try? url.resourceValues(forKeys: [.contentTypeKey]).contentType == .applicationBundle
                result = flag ?? false
            }

            // wait a bit for verification result
            _ = group.wait(timeout: .now() + 0.5)
        }
        return result
    }

    func performDrop(info: DropInfo) -> Bool {
        // handling code is here
        return true
    }
}